diff --git a/.github/workflows/maven-publish.yml b/.github/workflows/maven-publish.yml new file mode 100644 index 00000000..84f19cef --- /dev/null +++ b/.github/workflows/maven-publish.yml @@ -0,0 +1,34 @@ +# This workflow will build a package using Maven and then publish it to GitHub packages when a release is created +# For more information see: https://github.com/actions/setup-java/blob/main/docs/advanced-usage.md#apache-maven-with-a-settings-path + +name: Maven Package + +on: + release: + types: [created] + +jobs: + build: + + runs-on: ubuntu-latest + permissions: + contents: read + packages: write + + steps: + - uses: actions/checkout@v3 + - name: Set up JDK 17 + uses: actions/setup-java@v3 + with: + java-version: '17' + distribution: 'temurin' + server-id: github # Value of the distributionManagement/repository/id field of the pom.xml + settings-path: ${{ github.workspace }} # location for the settings.xml file + + - name: Build with Maven + run: mvn -B package --file pom.xml + + - name: Publish to GitHub Packages Apache Maven + run: mvn deploy -B -s $GITHUB_WORKSPACE/settings.xml + env: + GITHUB_TOKEN: ${{ github.token }} diff --git a/CHANGELOG.md b/CHANGELOG.md index a34cd87f..35936384 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,14 @@ # Changelog +## [4.0.0] - 2023-05-03 + +- Drop electron app +- Use javafx for the desktop app +- Re-written in kotlin +- Several enhancements + +### Changed + ## [3.5.4] - 2021-05-29 ### Changed diff --git a/README.md b/README.md index b19e39e7..753bc4de 100644 --- a/README.md +++ b/README.md @@ -1,35 +1,19 @@ # VRipper! -This is my spin for a cross platform gallery ripper app for [vipergirls](https://vipergirls.to) website. The purpose of -this project is to build a robust and clean application to conveniently download photo galleries using modern web -technologies, Java + Spring boot for the back end and angular + electron for the front end. +This is my spin for a cross-platform gallery ripper for [vipergirls.to](https://vipergirls.to). ## How to build -You need a recent version of maven 3.6.1+. +You need JDK 17 and a recent version of maven 3.6.1+ -The application support 2 build modes: +To build run the following: -- Desktop app: Self contained app, built with electron includes Java runtimes. -- Server app: Depends on Java runtimes, you need a web browser to access the app UI. + mvn clean install -Most people will be interested only on the desktop app, however if you have a nas or a server, the server app is there -for you. +Build artifact is located under -To build the server app: + vripper-project\vripper-gui\target\vripper-gui-4.0.0.jar - mvn clean install -DskipTests +Copy the artifact into any other folder and run: -To build the desktop app: - - mvn clean install -Pelectron -DskipTests - -Maven will automatically handle front end compilation. However, for development, you will need to install a recent -version of nodejs on your system. - -If you feel generous and want to support me, you can donate some coins, and I would be very grateful 🙏 - -I accept [PayPal](https://www.paypal.com/myaccount/transfer/homepage/buy/preview) donations, following is my email -address - - dev.vripper@gmail.com + java -jar vripper-gui-4.0.0.jar diff --git a/pom.xml b/pom.xml index 55f1b112..e6b1c7c9 100644 --- a/pom.xml +++ b/pom.xml @@ -1,20 +1,17 @@ - - 4.0.0 - tn.mnlr - vripper - 3.5.4 - pom - - org.springframework.boot - spring-boot-starter-parent - 2.4.5 - - - - vripper-server - vripper-ui - vripper-electron - + + 4.0.0 + me.mnlr + vripper-project + + vripper-core + vripper-gui + + pom + 4.0.0 + + true + diff --git a/vripper-server/.gitignore b/vripper-core/.gitignore similarity index 100% rename from vripper-server/.gitignore rename to vripper-core/.gitignore diff --git a/vripper-server/.mvn/wrapper/maven-wrapper.jar b/vripper-core/.mvn/wrapper/maven-wrapper.jar similarity index 100% rename from vripper-server/.mvn/wrapper/maven-wrapper.jar rename to vripper-core/.mvn/wrapper/maven-wrapper.jar diff --git a/vripper-server/.mvn/wrapper/maven-wrapper.properties b/vripper-core/.mvn/wrapper/maven-wrapper.properties similarity index 100% rename from vripper-server/.mvn/wrapper/maven-wrapper.properties rename to vripper-core/.mvn/wrapper/maven-wrapper.properties diff --git a/vripper-server/mvnw b/vripper-core/mvnw similarity index 100% rename from vripper-server/mvnw rename to vripper-core/mvnw diff --git a/vripper-server/mvnw.cmd b/vripper-core/mvnw.cmd similarity index 100% rename from vripper-server/mvnw.cmd rename to vripper-core/mvnw.cmd diff --git a/vripper-core/pom.xml b/vripper-core/pom.xml new file mode 100644 index 00000000..70c4231b --- /dev/null +++ b/vripper-core/pom.xml @@ -0,0 +1,128 @@ + + + 4.0.0 + + spring-boot-starter-parent + org.springframework.boot + + 3.0.6 + + me.mnlr.vripper + vripper-core + 4.0.0 + vripper-core + vripper-core + + 17 + 1.8.21 + false + + + + + kotlin-reflect + org.jetbrains.kotlin + + + kotlin-stdlib + org.jetbrains.kotlin + + + jackson-module-kotlin + com.fasterxml.jackson.module + + + spring-boot-starter-jdbc + org.springframework.boot + + + liquibase-core + org.liquibase + + + hsqldb + org.hsqldb + runtime + + + httpclient + org.apache.httpcomponents + + + htmlcleaner + net.sourceforge.htmlcleaner + 2.26 + + + reactor-core + io.projectreactor + + + failsafe + net.jodah + 2.4.4 + + + caffeine + com.github.ben-manes.caffeine + + + jackson-databind + com.fasterxml.jackson.core + + + com.fasterxml.jackson.dataformat + jackson-dataformat-yaml + + + com.fasterxml.jackson.datatype + jackson-datatype-jsr310 + + + + spring-boot-starter-test + org.springframework.boot + test + + + spring-restdocs-mockmvc + org.springframework.restdocs + test + + + + + ${project.basedir}/src/main/kotlin + ${project.basedir}/src/test/kotlin + + + org.jetbrains.kotlin + kotlin-maven-plugin + + + -Xjsr305=strict + + + spring + + + + + kotlin-maven-allopen + org.jetbrains.kotlin + ${kotlin.version} + + + + + + + + github + GitHub death-claw Apache Maven Packages + https://maven.pkg.github.com/death-claw/vripper-project + + + diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/AppEndpointService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/AppEndpointService.kt new file mode 100644 index 00000000..6d6cb394 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/AppEndpointService.kt @@ -0,0 +1,111 @@ +package me.mnlr.vripper + +import org.springframework.stereotype.Service +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.DownloadService +import me.mnlr.vripper.exception.PostParseException +import me.mnlr.vripper.model.PostItem +import me.mnlr.vripper.repositories.LogEventRepository +import me.mnlr.vripper.repositories.ThreadRepository +import me.mnlr.vripper.services.* +import me.mnlr.vripper.tasks.LinkScanRunnable +import java.util.* +import java.util.regex.Pattern + +@Service +class AppEndpointService( + private val downloadService: DownloadService, + private val dataTransaction: DataTransaction, + private val threadRepository: ThreadRepository, + private val threadCacheService: ThreadCacheService, + private val eventRepository: LogEventRepository, + private val threadPoolService: ThreadPoolService, +) { + private val log by LoggerDelegate() + + @Synchronized + fun scanLinks(postLinks: String) { + if (postLinks.isBlank()) { + log.warn("Nothing to scan") + return + } + val urlList = postLinks.split(Pattern.compile("\\r?\\n")).dropLastWhile { it.isEmpty() }.map { it.trim() } + .filter { it.isNotEmpty() } + threadPoolService.generalExecutor.submit(LinkScanRunnable(urlList)) + } + + @Synchronized + fun restartAll(posIds: List = listOf()) { + downloadService.restartAll(posIds) + } + + @Synchronized + fun download(posts: List>) { + downloadService.download(posts) + } + + @Synchronized + fun stopAll(postIdList: List?) { + downloadService.stopAll(postIdList) + } + + @Synchronized + fun remove(postIdList: List) { + downloadService.stopAll(postIdList) + dataTransaction.removeAll(postIdList) + } + + @Synchronized + fun clearCompleted(): List { + return dataTransaction.clearCompleted() + } + + @Synchronized + @Throws(PostParseException::class) + fun grab(threadId: String): List { + return try { + val thread = threadRepository.findByThreadId(threadId).orElseThrow { + PostParseException( + String.format( + "Unable to find links for threadId = %s", threadId + ) + ) + } + val threadLookupResult = threadCacheService[thread.threadId] + threadLookupResult.postItemList.ifEmpty { + log.error( + String.format( + "Failed to get links for threadId = %s", threadId + ) + ) + throw PostParseException( + String.format( + "Failed to get links for threadId = %s", threadId + ) + ) + } + } catch (e: Exception) { + throw PostParseException( + String.format( + "Failed to get links for threadId = %s, %s", threadId, e.message + ) + ) + } + } + + @Synchronized + fun threadRemove(threadIdList: List) { + threadIdList.forEach { + dataTransaction.removeThread(it) + } + } + + @Synchronized + fun threadClear() { + dataTransaction.clearQueueLinks() + } + + fun logClear() { + eventRepository.deleteAll() + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/EventListenerBean.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/EventListenerBean.kt new file mode 100644 index 00000000..05ec0572 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/EventListenerBean.kt @@ -0,0 +1,19 @@ +package me.mnlr.vripper + +import org.springframework.context.event.ContextRefreshedEvent +import org.springframework.context.event.EventListener +import org.springframework.stereotype.Component +import me.mnlr.vripper.repositories.PostDownloadStateRepository +import me.mnlr.vripper.services.DataTransaction + +@Component +class EventListenerBean( + private val postDownloadStateRepository: PostDownloadStateRepository, + private val dataTransaction: DataTransaction +) { + @EventListener + fun onApplicationEvent(event: ContextRefreshedEvent?) { + postDownloadStateRepository.setDownloadingToStopped() + dataTransaction.sortPostsByRank() + } +} diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/Management.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/Management.kt new file mode 100644 index 00000000..d9720ad9 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/Management.kt @@ -0,0 +1,58 @@ +package me.mnlr.vripper + +import jakarta.annotation.PreDestroy +import org.springframework.beans.factory.annotation.Value +import org.springframework.jdbc.core.JdbcTemplate +import org.springframework.stereotype.Component +import me.mnlr.vripper.download.DownloadService +import me.mnlr.vripper.services.ThreadPoolService +import java.nio.file.Path +import java.time.format.DateTimeFormatter +import java.util.* +import java.util.regex.Pattern + +@Component +class Management( + private val jdbcTemplate: JdbcTemplate, + @Value("\${base.dir}") baseDir: String, + @Value("\${base.dir.name}") baseDirName: String, + private val threadPoolService: ThreadPoolService, + private val downloadService: DownloadService +) { + private val backupFolder: Path + + init { + backupFolder = Path.of(baseDir, baseDirName, "backup") + } + + @PreDestroy + fun destroy() { + threadPoolService.destroy() + downloadService.destroy() + jdbcTemplate.execute("SHUTDOWN") + } + +// @PostConstruct +// @Scheduled(cron = "0 0 0 ? * *") +// private fun backup() { +// +// for (file in Files.list(backupFolder)) { +// val matcher = BACKUP_FILE_PATTERN.matcher(file.fileName.toString()) +// if (matcher.find()) { +// val localDate = LocalDate.parse(matcher.group(1), FORMATTER) +// if (localDate.isEqual(LocalDate.now())) { +// return +// } +// } +// } +// +// // create a backup file +// val backupFile = backupFolder.resolve("db_${LocalDateTime.now().format(FORMATTER)}.tar.gz") +// jdbcTemplate.execute("BACKUP DATABASE TO '${backupFile}' BLOCKING") +// } + + companion object { + val FORMATTER: DateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd") + private val BACKUP_FILE_PATTERN = Pattern.compile("^db_(\\d{4}-\\d{2}-\\d{2})\\.tar\\.gz$") + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/SpringContext.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/SpringContext.kt new file mode 100644 index 00000000..28b51890 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/SpringContext.kt @@ -0,0 +1,37 @@ +package me.mnlr.vripper + +import org.springframework.beans.BeansException +import org.springframework.context.ApplicationContext +import org.springframework.context.ApplicationContextAware +import org.springframework.context.ConfigurableApplicationContext +import org.springframework.scheduling.annotation.EnableScheduling +import org.springframework.stereotype.Component +import me.mnlr.vripper.delegate.LoggerDelegate + +@Component +@EnableScheduling +class SpringContext : ApplicationContextAware { + @Throws(BeansException::class) + override fun setApplicationContext(context: ApplicationContext) { + + // store ApplicationContext reference to access required beans later on + Companion.context = context as ConfigurableApplicationContext + } + + companion object { + private val log by LoggerDelegate() + private lateinit var context: ConfigurableApplicationContext + fun getBean(beanClass: Class): T { + return context.getBean(beanClass) + } + + fun getBeansOfType(beanClass: Class): Map { + return context.getBeansOfType(beanClass) + } + + fun close() { + log.info("Application terminating...") + context.close() + } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/Utils.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/Utils.kt new file mode 100644 index 00000000..1c98dca9 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/Utils.kt @@ -0,0 +1,39 @@ +package me.mnlr.vripper + +import java.io.PrintWriter +import java.io.StringWriter + +fun Throwable.formatToString(): String { + return StringWriter().use { stringWriter -> + PrintWriter(stringWriter).use { printWriter -> + this.printStackTrace(printWriter) + printWriter.flush() + stringWriter.flush() + stringWriter.toString() + } + } +} + +fun Long.formatSI(): String { + return humanReadableByteCount(this, false) +} + +private fun humanReadableByteCount(bytes: Long, si: Boolean): String { + val unit = if (si) 1000 else 1024 + if (bytes < unit) return "$bytes B" + val exp = (Math.log(bytes.toDouble()) / Math.log(unit.toDouble())).toInt() + val pre = (if (si) "kMGTPE" else "KMGTPE")[exp - 1].toString() + if (si) "" else "i" + return String.format("%.1f %sB", bytes / Math.pow(unit.toDouble(), exp.toDouble()), pre) +} + + +fun getExtension(fileName: String): String { + return if (fileName.contains(".")) fileName.substring(fileName.lastIndexOf(".") + 1) else "" +} + +fun getFileNameWithoutExtension(fileName: String): String { + return if (fileName.contains(".")) fileName.substring( + 0, + fileName.lastIndexOf(".") + ) else fileName +} diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/delegate/LoggerDelegate.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/delegate/LoggerDelegate.kt new file mode 100644 index 00000000..22a8df6d --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/delegate/LoggerDelegate.kt @@ -0,0 +1,16 @@ +package me.mnlr.vripper.delegate + +import org.slf4j.Logger +import org.slf4j.LoggerFactory +import kotlin.properties.ReadOnlyProperty +import kotlin.reflect.KProperty + +class LoggerDelegate : ReadOnlyProperty { + private var logger: Logger? = null + override operator fun getValue(thisRef: Any?, property: KProperty<*>): Logger { + if (logger == null) { + logger = LoggerFactory.getLogger(thisRef!!.javaClass) + } + return logger!! + } +} diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/download/DownloadService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/download/DownloadService.kt new file mode 100644 index 00000000..e0274154 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/download/DownloadService.kt @@ -0,0 +1,297 @@ +package me.mnlr.vripper.download + +import jakarta.annotation.PostConstruct +import net.jodah.failsafe.Failsafe +import net.jodah.failsafe.RetryPolicy +import org.springframework.beans.factory.annotation.Value +import org.springframework.stereotype.Service +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.entities.ImageDownloadState +import me.mnlr.vripper.entities.LogEvent +import me.mnlr.vripper.entities.PostDownloadState +import me.mnlr.vripper.entities.domain.Status +import me.mnlr.vripper.formatToString +import me.mnlr.vripper.host.Host +import me.mnlr.vripper.repositories.ImageRepository +import me.mnlr.vripper.repositories.LogEventRepository +import me.mnlr.vripper.repositories.PostDownloadStateRepository +import me.mnlr.vripper.services.* +import java.util.* +import java.util.concurrent.ExecutorService +import java.util.concurrent.Executors +import java.util.concurrent.TimeUnit +import java.util.concurrent.locks.ReentrantLock +import java.util.stream.Collectors +import kotlin.concurrent.withLock + +@Service +class DownloadService( + @param:Value("\${download.pool-size}") private val maxPoolSize: Int, + private val settingsService: SettingsService, + private val dataTransaction: DataTransaction, + private val metadataService: MetadataService, + private val hosts: List, + private val eventRepository: LogEventRepository, + private val retryPolicyService: RetryPolicyService, + private val postDownloadStateRepository: PostDownloadStateRepository, + private val imageRepository: ImageRepository, + private val threadPoolService: ThreadPoolService, +) { + private val log by LoggerDelegate() + + // Class fields + private val executor: ExecutorService = Executors.newFixedThreadPool(maxPoolSize) + private val running: MutableMap> = mutableMapOf() + private val pending: MutableMap> = mutableMapOf() + private val lock = ReentrantLock() + private val condition = lock.newCondition() + private val pollThread: Thread + + init { + pollThread = Thread( + { + val accepted: MutableList = mutableListOf() + val candidates: MutableList = mutableListOf() + while (!Thread.interrupted()) { + lock.withLock { + candidates.addAll(getCandidates(candidateCount())) + candidates.forEach { + if (canRun(it.context.image.host)) { + accepted.add(it) + running[it.context.image.host]!!.add(it) + log.debug("${it.context.image.url} accepted to run") + } + } + accepted.forEach { + pending[it.context.image.host]?.remove(it) + schedule(it) + } + accepted.clear() + candidates.clear() + condition.await() + } + } + }, "Download scheduler thread" + ) + } + + @PostConstruct + fun init() { + pollThread.start() + } + + fun destroy() { + log.info("Shutting down ExecutionService") + pollThread.interrupt() + executor.shutdown() + stop(postDownloadStateRepository.findAll().map { it.postId }) + if (!executor.awaitTermination(5, TimeUnit.SECONDS)) { + log.warn("Some jobs are still running!, forcing shutdown") + executor.shutdownNow() + } + } + + fun stopAll(postIds: List?) { + if (postIds != null) { + stop(postIds) + } else { + stop(postDownloadStateRepository.findAll().map(PostDownloadState::postId)) + } + } + + fun restartAll(postIds: List = listOf()) { + if (postIds.isNotEmpty()) { + restart(postIds) + } else { + restart(postDownloadStateRepository.findAll().map(PostDownloadState::postId)) + } + } + + private fun restart(postIds: List) { + lock.withLock { + val data: MutableMap> = mutableMapOf() + for (postId in postIds) { + if (isPending(postId)) { + log.warn("Cannot restart, jobs are currently running for post id $postIds") + continue + } + val imageDownloadStates: List = + imageRepository.findByPostIdAndIsNotCompleted(postId) + if (imageDownloadStates.isEmpty()) { + continue + } + val postDownloadState: PostDownloadState = + postDownloadStateRepository.findByPostId(postId).orElseThrow() + log.debug("Restarting ${imageDownloadStates.size} jobs for post id $postIds") + postDownloadState.status = Status.PENDING + dataTransaction.update(postDownloadState) + data[postDownloadState] = imageDownloadStates + } + + for ((postDownloadState, imageDownloadStates) in data) { + postDownloadState.status = Status.PENDING + dataTransaction.update(postDownloadState) + for (imageDownloadState in imageDownloadStates) { + log.debug("Enqueuing a job for ${imageDownloadState.url}") + with(imageDownloadState) { + this.status = Status.STOPPED + this.current = 0 + } + dataTransaction.update(imageDownloadState) + val imageDownloadRunnable = ImageDownloadRunnable( + imageDownloadState.id!!, settingsService.settings + ) + pending.computeIfAbsent( + imageDownloadState.host + ) { mutableListOf() } + pending[imageDownloadState.host]!!.add(imageDownloadRunnable) + } + } + condition.signal() + } + } + + private fun isPending(postId: String): Boolean { + lock.withLock { + return pending.values.flatten().any { it.context.image.postId == postId } + } + } + + private fun isRunning(postId: String): Boolean { + lock.withLock { + return running.values.flatten().any { it.context.image.postId == postId } + } + } + + private fun stop(postIds: List) { + lock.withLock { + for (postId in postIds) { + val postDownloadState: PostDownloadState = + postDownloadStateRepository.findByPostId(postId).orElseThrow() + pending.values.forEach { pending -> + pending.removeIf { it.context.image.postId == postId } + } + running.values.flatten() + .filter { p: ImageDownloadRunnable -> p.context.image.postId == postId } + .forEach { obj: ImageDownloadRunnable -> obj.stop() } + dataTransaction.stopImagesByPostIdAndIsNotCompleted(postId) + dataTransaction.finishPost(postDownloadState) + } + metadataService.stopFetchingMetadata(postIds) + } + } + + private fun canRun(host: Host): Boolean { + val totalRunning = running.values.sumOf { it.size } + return (running[host]!!.size < settingsService.settings.connectionSettings.maxThreads && if (settingsService.settings.connectionSettings.maxTotalThreads == 0) totalRunning < maxPoolSize else totalRunning < settingsService.settings.connectionSettings.maxTotalThreads) + } + + private fun candidateCount(): Map { + val map: MutableMap = mutableMapOf() + hosts.forEach { h: Host -> + val imageDownloadRunnableList: List = running.computeIfAbsent( + h + ) { mutableListOf() } + val count: Int = settingsService.settings.connectionSettings.maxThreads - imageDownloadRunnableList.size + log.debug("Download slots for ${h.host}: $count") + map[h] = count + } + return map + } + + private fun getCandidates(candidateCount: Map): List { + val hostIntegerMap: MutableMap = candidateCount.toMutableMap() + val candidates: MutableList = mutableListOf() + hosts@ for (host in pending.keys) { + val list: List = + pending[host]!!.sortedWith(Comparator.comparingInt { it.context.post.rank } + .thenComparingInt { it.context.image.index }) + for (imageDownloadRunnable in list) { + val count = hostIntegerMap[host] ?: 0 + if (count > 0) { + candidates.add(imageDownloadRunnable) + hostIntegerMap[host] = count - 1 + } else { + continue@hosts + } + } + } + if (log.isDebugEnabled) { + val collect: Map> = + candidates.stream().collect(Collectors.groupingBy { it.context.image.host }) + collect.forEach { + log.debug( + "Candidate download for ${it.key.host} ${it.value.size}/${candidateCount[it.key]}" + ) + } + } + return candidates.sortedWith(Comparator.comparing { v: ImageDownloadRunnable -> v.context.post.rank }) + } + + private fun schedule(imageDownloadRunnable: ImageDownloadRunnable) { + log.debug("Scheduling a job for ${imageDownloadRunnable.context.image.url}") + executor.execute { + try { + Failsafe.with>(retryPolicyService.buildRetryPolicyForDownload()) + .onFailure { + try { + eventRepository.save( + LogEvent( + type = LogEvent.Type.DOWNLOAD, + status = LogEvent.Status.ERROR, + message = "Failed to download ${imageDownloadRunnable.context.image.url}\n ${it.failure.formatToString()}" + ) + ) + } catch (exp: Exception) { + log.error("Failed to save event", exp) + } + log.error( + "Failed to download ${imageDownloadRunnable.context.image.url} after ${it.attemptCount} tries", + it.failure + ) + val image = imageDownloadRunnable.context.image + image.status = Status.ERROR + dataTransaction.update(image) + }.onComplete { + + afterJobFinish(imageDownloadRunnable) + log.debug( + "Finished downloading ${imageDownloadRunnable.context.image.url}" + ) + }.run(imageDownloadRunnable::run) + } catch (ignored: Exception) { + } + } + } + + fun afterJobFinish(imageDownloadRunnable: ImageDownloadRunnable) { + lock.withLock { + running[imageDownloadRunnable.context.image.host]!!.remove(imageDownloadRunnable) + if (!isPending(imageDownloadRunnable.context.image.postId) && !isRunning( + imageDownloadRunnable.context.image.postId + ) + ) { + dataTransaction.finishPost(imageDownloadRunnable.context.post) + } + condition.signal() + } + } + + fun pendingCount(): Int { + return pending.values.sumOf { it.size } + } + + fun runningCount(): Int { + return running.values.sumOf { it.size } + } + + fun download(posts: List>) { + for (post in posts) { + threadPoolService.generalExecutor.submit( + PostDownloadRunnable( + post.first, post.second + ) + ) + } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/download/ImageDownloadContext.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/download/ImageDownloadContext.kt new file mode 100644 index 00000000..38a210bf --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/download/ImageDownloadContext.kt @@ -0,0 +1,25 @@ +package me.mnlr.vripper.download + +import org.apache.http.client.protocol.HttpClientContext +import me.mnlr.vripper.SpringContext +import me.mnlr.vripper.entities.ImageDownloadState +import me.mnlr.vripper.entities.PostDownloadState +import me.mnlr.vripper.repositories.ImageRepository +import me.mnlr.vripper.repositories.PostDownloadStateRepository + +class ImageDownloadContext(val imageId: Long) { + + private val imageRepository: ImageRepository = + SpringContext.getBean(ImageRepository::class.java) + private val postDownloadStateRepository: PostDownloadStateRepository = + SpringContext.getBean(PostDownloadStateRepository::class.java) + + + val httpContext: HttpClientContext = HttpClientContext.create() + val postId = image.postIdRef + var stopped = false + val image: ImageDownloadState + get() = imageRepository.findById(imageId).orElseThrow() + val post: PostDownloadState + get() = postDownloadStateRepository.findById(postId).orElseThrow() +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/download/ImageDownloadRunnable.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/download/ImageDownloadRunnable.kt new file mode 100644 index 00000000..9dab397a --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/download/ImageDownloadRunnable.kt @@ -0,0 +1,178 @@ +package me.mnlr.vripper.download + +import net.jodah.failsafe.function.CheckedRunnable +import org.apache.http.client.protocol.HttpClientContext +import org.apache.http.impl.client.BasicCookieStore +import me.mnlr.vripper.SpringContext +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.entities.ImageDownloadState +import me.mnlr.vripper.entities.PostDownloadState +import me.mnlr.vripper.entities.domain.Status +import me.mnlr.vripper.exception.DownloadException +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.getExtension +import me.mnlr.vripper.host.DownloadedImage +import me.mnlr.vripper.host.ImageMimeType +import me.mnlr.vripper.model.Settings +import me.mnlr.vripper.services.* +import java.io.IOException +import java.nio.file.Files +import java.nio.file.Path +import java.nio.file.StandardCopyOption +import java.util.* +import kotlin.io.path.pathString + +class ImageDownloadRunnable( + private val imageInternalId: Long, private val settings: Settings +) : CheckedRunnable { + private val log by LoggerDelegate() + + private val dataTransaction: DataTransaction = SpringContext.getBean(DataTransaction::class.java) + private val pathService: PathService = SpringContext.getBean(PathService::class.java) + + val context: ImageDownloadContext = ImageDownloadContext(imageInternalId) + private val image: ImageDownloadState + get() = context.image + private var stopped: Boolean + get() = context.stopped + set(value) { + context.stopped = value + } + private val httpContext: HttpClientContext + get() = context.httpContext + + init { + httpContext.cookieStore = BasicCookieStore() + httpContext.setAttribute( + HTTPService.ContextAttributes.CONTEXT_ATTRIBUTES, HTTPService.ContextAttributes() + ) + } + + @Throws(DownloadException::class) + fun download() { + try { + val image = image + with(image) { + this.status = Status.DOWNLOADING + this.current = 0 + } + dataTransaction.update(image) + synchronized(image.postId.intern()) { + val post = context.post + if (post.status != Status.DOWNLOADING) { + post.status = Status.DOWNLOADING + dataTransaction.update(post) + } + } + + if (stopped) { + return + } + log.debug("Getting image url and name from ${image.url} using ${image.host}") + val downloadedImage = image.host.downloadInternal(image.url, context) + log.debug("Resolved name for ${image.url}: ${downloadedImage.name}") + log.debug( + "Downloaded image ${image.url} to ${downloadedImage.path}" + ) + val sanitizedFileName = pathService.sanitize(downloadedImage.name) + log.debug( + "Sanitizing image name from ${downloadedImage.name} to $sanitizedFileName" + ) + checkImageTypeAndRename( + context.post, downloadedImage, image.index + ) + synchronized(image.postId.intern()) { + val post = context.post + post.done += 1 + dataTransaction.update(post) + } + } catch (e: Exception) { + if (stopped) { + return + } + throw DownloadException(e) + } finally { + val image = image + if (image.current == image.total && image.total > 0) { + image.status = Status.FINISHED + } else if (stopped) { + image.status = Status.STOPPED + } else { + image.status = Status.ERROR + } + dataTransaction.update(image) + } + } + + @Throws(HostException::class) + private fun checkImageTypeAndRename( + postDownloadState: PostDownloadState, downloadedImage: DownloadedImage, index: Int + ) { + val existingExtension = getExtension(downloadedImage.name) + val extension = when (downloadedImage.type) { + ImageMimeType.IMAGE_BMP -> "BMP" + ImageMimeType.IMAGE_GIF -> "GIF" + ImageMimeType.IMAGE_JPEG -> "JPG" + ImageMimeType.IMAGE_PNG -> "PNG" + ImageMimeType.IMAGE_WEBP -> "WEBP" + } + val filename = if(existingExtension.isBlank()) "${downloadedImage.name}.$extension" else downloadedImage.name + try { + val downloadDestinationFolder = Path.of(postDownloadState.downloadDirectory) + synchronized(downloadDestinationFolder.pathString.intern()) { + Files.createDirectories(downloadDestinationFolder) + } + val image = downloadDestinationFolder.resolve( + "${ + if (settings.downloadSettings.forceOrder) String.format( + "%03d_", index + 1 + ) else "" + }$filename" + ) + Files.copy(downloadedImage.path, image, StandardCopyOption.REPLACE_EXISTING) + } catch (e: Exception) { + throw HostException("Failed to rename the image", e) + } finally { + try { + Files.delete(downloadedImage.path) + } catch (e: IOException) { + log.warn( + "Failed to delete temporary file ${downloadedImage.path}" + ) + } + } + } + + @Throws(Exception::class) + override fun run() { + if (stopped) { + return + } + download() + } + + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (other == null || javaClass != other.javaClass) return false + val that = other as ImageDownloadRunnable + return imageInternalId == that.imageInternalId + } + + override fun hashCode(): Int { + return Objects.hash(imageInternalId) + } + + fun stop() { + stopped = true + val contextAttributes = httpContext.getAttribute( + HTTPService.ContextAttributes.CONTEXT_ATTRIBUTES, HTTPService.ContextAttributes::class.java + ) + if (contextAttributes != null) { + synchronized(contextAttributes.requests) { + for (request in contextAttributes.requests) { + request.abort() + } + } + } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/download/PostDownloadRunnable.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/download/PostDownloadRunnable.kt new file mode 100644 index 00000000..7f4f4bfa --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/download/PostDownloadRunnable.kt @@ -0,0 +1,162 @@ +package me.mnlr.vripper.download + +import net.jodah.failsafe.Failsafe +import net.jodah.failsafe.function.CheckedSupplier +import org.apache.http.client.HttpClient +import org.apache.http.client.methods.CloseableHttpResponse +import org.apache.http.client.methods.HttpGet +import org.apache.http.client.protocol.HttpClientContext +import org.apache.http.client.utils.URIBuilder +import org.apache.http.util.EntityUtils +import me.mnlr.vripper.SpringContext +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.entities.LogEvent +import me.mnlr.vripper.entities.LogEvent.Status.* +import me.mnlr.vripper.exception.DownloadException +import me.mnlr.vripper.exception.PostParseException +import me.mnlr.vripper.formatToString +import me.mnlr.vripper.model.PostItem +import me.mnlr.vripper.parser.ThreadLookupAPIResponseHandler +import me.mnlr.vripper.repositories.LogEventRepository +import me.mnlr.vripper.services.* +import java.net.URISyntaxException +import java.util.* +import javax.xml.parsers.SAXParserFactory + +class PostDownloadRunnable(private val threadId: String, private val postId: String) : Runnable { + private val log by LoggerDelegate() + private val dataTransaction: DataTransaction = + SpringContext.getBean(DataTransaction::class.java) + private val settingsService: SettingsService = + SpringContext.getBean(SettingsService::class.java) + private val vgAuthService: VGAuthService = SpringContext.getBean(VGAuthService::class.java) + private val eventRepository: LogEventRepository = + SpringContext.getBean(LogEventRepository::class.java) + private val cm: HTTPService = SpringContext.getBean(HTTPService::class.java) + private val retryPolicyService: RetryPolicyService = + SpringContext.getBean(RetryPolicyService::class.java) + private val downloadService: DownloadService = + SpringContext.getBean(DownloadService::class.java) + private val link: String = + "${settingsService.settings.viperSettings.host}/threads/$threadId?p=$postId" + private val logEvent: LogEvent + + init { + logEvent = eventRepository.save( + LogEvent( + type = LogEvent.Type.POST, + status = PENDING, + message = "Processing $link" + ) + ) + } + + override fun run() { + try { + eventRepository.update(logEvent.copy(status = PROCESSING)) + if (dataTransaction.exists(postId)) { + log.warn(String.format("skipping %s, already loaded", postId)) + eventRepository.update( + logEvent.copy( + status = ERROR, + message = String.format("Gallery %s is already loaded", link) + ) + ) + return + } + val postItem: PostItem = try { + parse() + } catch (e: PostParseException) { + val error = String.format("parsing failed for gallery %s", link) + log.error(error, e) + eventRepository.update( + logEvent.copy( + status = ERROR, message = """ + $error + ${e.formatToString()} + """.trimIndent() + ) + ) + return + } + if (postItem.imageItemList.isEmpty()) { + val error = String.format("Gallery %s contains no images to download", link) + log.error(error) + eventRepository.update(logEvent.copy(status = ERROR, message = error)) + return + } + val post = dataTransaction.newPost(postItem) + vgAuthService.leaveThanks(post) +// metadataService.startFetchingMetadata(post) + if (settingsService.settings.downloadSettings.autoStart) { + log.debug("Auto start downloads option is enabled") + downloadService.restartAll(listOf(postItem.postId)) + log.debug(String.format("Done enqueuing jobs for %s", postItem.url)) + } + + eventRepository.update( + logEvent.copy( + status = DONE, + message = String.format( + "Gallery %s is successfully added to download queue", + link + ) + ) + ) + } catch (e: Exception) { + val error = String.format("Error when adding gallery %s", link) + log.error(error, e) + eventRepository.update( + logEvent.copy( + status = ERROR, message = """ + $error + ${e.formatToString()} + """.trimIndent() + ) + ) + } + } + + @Throws(PostParseException::class) + fun parse(): PostItem { + log.debug("Parsing post $postId") + val httpGet: HttpGet = try { + val uriBuilder = URIBuilder("${settingsService.settings.viperSettings.host}/vr.php") + uriBuilder.setParameter("p", postId) + cm.buildHttpGet(uriBuilder.build(), HttpClientContext.create()) + } catch (e: URISyntaxException) { + throw PostParseException(e) + } + val threadLookupAPIResponseHandler = ThreadLookupAPIResponseHandler() + log.debug("Requesting $httpGet") + return try { + Failsafe.with(retryPolicyService.buildGenericRetryPolicy()).onFailure { + log.error( + "parsing failed for thread $threadId, post $postId", it.failure + ) + }.get(CheckedSupplier { + val connection: HttpClient = cm.client.build() + (connection.execute( + httpGet, vgAuthService.context + ) as CloseableHttpResponse).use { response -> + try { + if (response.statusLine.statusCode / 100 != 2) { + throw DownloadException("Unexpected response code '${response.statusLine.statusCode}' for $httpGet") + } + factory.newSAXParser() + .parse(response.entity.content, threadLookupAPIResponseHandler) + threadLookupAPIResponseHandler.result.postItemList[0] + } finally { + EntityUtils.consumeQuietly(response.entity) + } + } + }) + } catch (e: Exception) { + throw PostParseException(e) + } + } + + companion object { + private val factory = SAXParserFactory.newInstance() + } +} diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/ImageDownloadState.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/ImageDownloadState.kt new file mode 100644 index 00000000..6ccbf9e5 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/ImageDownloadState.kt @@ -0,0 +1,31 @@ +package me.mnlr.vripper.entities + +import me.mnlr.vripper.entities.domain.Status +import me.mnlr.vripper.host.Host + +data class ImageDownloadState( + var id: Long? = null, + val postId: String, + val url: String, + val host: Host, + val index: Int, + val postIdRef: Long, + var total: Long = -1, + var current: Long = 0, + var status: Status = Status.STOPPED, +) { + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (javaClass != other?.javaClass) return false + + other as ImageDownloadState + + if (url != other.url) return false + + return true + } + + override fun hashCode(): Int { + return url.hashCode() + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/LogEvent.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/LogEvent.kt new file mode 100644 index 00000000..b04e52c1 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/LogEvent.kt @@ -0,0 +1,22 @@ +package me.mnlr.vripper.entities + +import java.time.LocalDateTime + +data class LogEvent( + val id: Long? = null, + val type: Type, + val status: Status, + val time: LocalDateTime = LocalDateTime.now(), + val message: String, +) { + + enum class Type(val stringValue: String) { + POST("Post"), THREAD("Thread"), THANKS("Thanks"), METADATA("Metadata"), SCAN("Scan"), DOWNLOAD( + "Download" + ) + } + + enum class Status(val stringValue: String) { + PENDING("Pending"), PROCESSING("Processing"), DONE("Done"), ERROR("Error") + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/Metadata.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/Metadata.kt new file mode 100644 index 00000000..4b2ca78e --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/Metadata.kt @@ -0,0 +1,19 @@ +package me.mnlr.vripper.entities + +class Metadata { + var postIdRef: Long? = null + var postId: String? = null + var postedBy: String? = null + var resolvedNames = emptyList() + + companion object { + fun from(metadata: Metadata): Metadata { + val copy = Metadata() + copy.postIdRef = metadata.postIdRef + copy.postId = metadata.postId + copy.postedBy = metadata.postedBy + copy.resolvedNames = metadata.resolvedNames + return copy + } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/PostDownloadState.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/PostDownloadState.kt new file mode 100644 index 00000000..80cd4727 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/PostDownloadState.kt @@ -0,0 +1,37 @@ +package me.mnlr.vripper.entities + +import me.mnlr.vripper.entities.domain.Status +import java.time.LocalDateTime + +data class PostDownloadState( + var id: Long? = null, + val postTitle: String, + val threadTitle: String, + val forum: String, + val url: String, + val token: String, + val postId: String, + val threadId: String, + val total: Int, + val hosts: Set, + val downloadDirectory: String, + val addedOn: LocalDateTime = LocalDateTime.now(), + var status: Status = Status.STOPPED, + var done: Int = 0, + var rank: Int = Int.MAX_VALUE +) { + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (javaClass != other?.javaClass) return false + + other as PostDownloadState + + if (postId != other.postId) return false + + return true + } + + override fun hashCode(): Int { + return postId.hashCode() + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/Thread.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/Thread.kt new file mode 100644 index 00000000..ae9e35e4 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/Thread.kt @@ -0,0 +1,21 @@ +package me.mnlr.vripper.entities + +data class Thread( + val id: Long? = null, + val link: String, + val threadId: String, + var total: Int = 0, +) { + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (javaClass != other?.javaClass) return false + + other as Thread + + return threadId == other.threadId + } + + override fun hashCode(): Int { + return threadId.hashCode() + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/domain/Status.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/domain/Status.kt new file mode 100644 index 00000000..aed4a6f9 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/entities/domain/Status.kt @@ -0,0 +1,5 @@ +package me.mnlr.vripper.entities.domain + +enum class Status(val stringValue: String) { + PENDING("Pending"), DOWNLOADING("Downloading"), FINISHED("Finished"), ERROR("Error"), STOPPED("Stopped") +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/event/Event.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/event/Event.kt new file mode 100644 index 00000000..74da2272 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/event/Event.kt @@ -0,0 +1,7 @@ +package me.mnlr.vripper.event + +data class Event(val kind: Kind, val data: T) { + enum class Kind { + POST_UPDATE, POST_REMOVE, IMAGE_UPDATE, METADATA_UPDATE, THREAD_UPDATE, THREAD_REMOVE, LOG_EVENT_UPDATE, LOG_EVENT_REMOVE, VG_USER, DOWNLOAD_STATUS, BYTES_PER_SECOND, SETTINGS_UPDATE + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/event/EventBus.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/event/EventBus.kt new file mode 100644 index 00000000..164c7072 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/event/EventBus.kt @@ -0,0 +1,30 @@ +package me.mnlr.vripper.event + +import jakarta.annotation.PreDestroy +import org.springframework.stereotype.Service +import reactor.core.publisher.Flux +import reactor.core.publisher.SignalType +import reactor.core.publisher.Sinks +import reactor.core.publisher.Sinks.EmitFailureHandler +import reactor.core.publisher.Sinks.EmitResult + +@Service +class EventBus { + private val sink = Sinks.many().multicast().onBackpressureBuffer>() + fun publishEvent(event: Event<*>) { + sink.emitNext(event, RETRY) + } + + fun flux(): Flux> { + return sink.asFlux() + } + + @PreDestroy + private fun destroy() { + sink.emitComplete(RETRY) + } + + companion object { + val RETRY = EmitFailureHandler { _: SignalType?, _: EmitResult? -> true } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/DownloadException.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/DownloadException.kt new file mode 100644 index 00000000..d64ebe72 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/DownloadException.kt @@ -0,0 +1,6 @@ +package me.mnlr.vripper.exception + +class DownloadException : Exception { + constructor(message: String?) : super(message) + constructor(e: Throwable?) : super(e) +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/HostException.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/HostException.kt new file mode 100644 index 00000000..3d0e18ce --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/HostException.kt @@ -0,0 +1,7 @@ +package me.mnlr.vripper.exception + +class HostException : Exception { + constructor(e: Throwable?) : super(e) + constructor(message: String?) : super(message) + constructor(message: String?, e: Throwable?) : super(message, e) +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/HtmlProcessorException.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/HtmlProcessorException.kt new file mode 100644 index 00000000..aa7da06d --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/HtmlProcessorException.kt @@ -0,0 +1,3 @@ +package me.mnlr.vripper.exception + +class HtmlProcessorException(e: Throwable?) : Exception(e) \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/PostParseException.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/PostParseException.kt new file mode 100644 index 00000000..ab73809a --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/PostParseException.kt @@ -0,0 +1,7 @@ +package me.mnlr.vripper.exception + +class PostParseException : Exception { + constructor(message: String?) : super(message) + constructor(message: String?, e: Throwable?) : super(message, e) + constructor(e: Throwable?) : super(e) +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/QueueException.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/QueueException.kt new file mode 100644 index 00000000..db37cb5f --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/QueueException.kt @@ -0,0 +1,7 @@ +package me.mnlr.vripper.exception + +class QueueException : Exception { + constructor(message: String?) : super(message) + constructor(message: String?, e: Throwable?) : super(message, e) + constructor(e: Throwable?) : super(e) +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/RenameException.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/RenameException.kt new file mode 100644 index 00000000..888de728 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/RenameException.kt @@ -0,0 +1,3 @@ +package me.mnlr.vripper.exception + +class RenameException(message: String?, e: Exception?) : Exception(message, e) \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/ValidationException.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/ValidationException.kt new file mode 100644 index 00000000..38197662 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/ValidationException.kt @@ -0,0 +1,3 @@ +package me.mnlr.vripper.exception + +class ValidationException(message: String) : Exception(message) \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/VripperException.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/VripperException.kt new file mode 100644 index 00000000..d57cae2c --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/VripperException.kt @@ -0,0 +1,6 @@ +package me.mnlr.vripper.exception + +class VripperException : Exception { + constructor(message: String?) : super(message) + constructor(e: Throwable?) : super(e) +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/XpathException.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/XpathException.kt new file mode 100644 index 00000000..1dbb6432 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/exception/XpathException.kt @@ -0,0 +1,3 @@ +package me.mnlr.vripper.exception + +class XpathException(e: Throwable?) : Exception(e) \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/AcidimgHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/AcidimgHost.kt new file mode 100644 index 00000000..69dd8c54 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/AcidimgHost.kt @@ -0,0 +1,104 @@ +package me.mnlr.vripper.host + +import org.apache.http.NameValuePair +import org.apache.http.client.HttpClient +import org.apache.http.client.entity.UrlEncodedFormEntity +import org.apache.http.client.methods.CloseableHttpResponse +import org.apache.http.message.BasicNameValuePair +import org.apache.http.util.EntityUtils +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* + +@Service +class AcidimgHost( + private val httpService: HTTPService, + private val htmlProcessorService: HtmlProcessorService, + private val xpathService: XpathService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + try { + log.debug( + String.format( + "Looking for xpath expression %s in %s", + CONTINUE_BUTTON_XPATH, + url + ) + ) + xpathService.getAsNode(document, CONTINUE_BUTTON_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } + log.debug(String.format("Click button found for %s", url)) + val client: HttpClient = httpService.client.build() + val httpPost = httpService.buildHttpPost(url, context.httpContext) + httpPost.addHeader("Referer", url) + val params: MutableList = ArrayList() + params.add(BasicNameValuePair("imgContinue", "Continue to your image")) + try { + httpPost.entity = UrlEncodedFormEntity(params) + } catch (e: Exception) { + throw HostException(e) + } + log.debug(String.format("Requesting %s", httpPost)) + val doc = try { + (client.execute( + httpPost, context.httpContext + ) as CloseableHttpResponse).use { response -> + log.debug(String.format("Cleaning response for %s", httpPost)) + try { + htmlProcessorService.clean(response.entity.content) + } finally { + EntityUtils.consumeQuietly(response.entity) + } + } + } catch (e: Exception) { + throw HostException(e) + } + val imgNode: Node = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(doc, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + val imgTitle = imgNode.attributes.getNamedItem("alt").textContent.trim() + val imgUrl = imgNode.attributes.getNamedItem("src").textContent.trim() + Pair( + imgTitle.ifEmpty { getDefaultImageName(imgUrl) }, imgUrl + ) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "acidimg.cc" + private const val lookup = "acidimg.cc" + private const val CONTINUE_BUTTON_XPATH = "//input[@id='continuebutton']" + private const val IMG_XPATH = "//img[@class='centred']" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/DPicMeHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/DPicMeHost.kt new file mode 100644 index 00000000..910b1a72 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/DPicMeHost.kt @@ -0,0 +1,69 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* +import java.util.* + +@Service +class DPicMeHost( + private val xpathService: XpathService, + httpService: HTTPService, + htmlProcessorService: HtmlProcessorService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val imgNode: Node = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(document, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + val imgTitle = + Optional.ofNullable(imgNode.attributes.getNamedItem("alt")) + .map { e: Node -> e.textContent.trim() } + .orElse("") + val imgUrl = + Optional.ofNullable(imgNode.attributes.getNamedItem("src")) + .map { e: Node -> e.textContent.trim() } + .orElse("") + var defaultName: String = UUID.randomUUID().toString() + val index = imgUrl.lastIndexOf('/') + if (index != -1 && index < imgUrl.length) { + defaultName = imgUrl.substring(imgUrl.lastIndexOf('/') + 1) + } + Pair((imgTitle.ifEmpty { defaultName })!!, imgUrl) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "dpic.me" + private const val lookup = "dpic.me" + private const val IMG_XPATH = "//img[@id='pic']" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/Host.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/Host.kt new file mode 100644 index 00000000..ae442a2d --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/Host.kt @@ -0,0 +1,229 @@ +package me.mnlr.vripper.host + +import org.apache.http.Header +import org.apache.http.client.HttpClient +import org.apache.http.client.methods.CloseableHttpResponse +import org.apache.http.client.protocol.HttpClientContext +import org.apache.http.util.EntityUtils +import org.w3c.dom.Document +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.DownloadException +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.getFileNameWithoutExtension +import me.mnlr.vripper.services.* +import java.io.FileOutputStream +import java.nio.file.Files +import java.nio.file.Path +import kotlin.io.path.pathString + +abstract class Host( + private val httpService: HTTPService, + private val htmlProcessorService: HtmlProcessorService, + private val dataTransaction: DataTransaction, + private val downloadSpeedService: DownloadSpeedService +) { + private val log by LoggerDelegate() + + companion object { + private const val READ_BUFFER_SIZE = 8192 + } + + abstract val host: String + + @Throws(HostException::class) + abstract fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair + + @Throws(HostException::class) + fun downloadInternal(url: String, context: ImageDownloadContext): DownloadedImage { + val headers = head(url, context.httpContext) + // is the body of type image ? + val imageMimeType = getImageMimeType(headers) + + val downloadedImage = if (imageMimeType != null) { + // a direct link, awesome + val downloadedImage = fetch(url, context) { + handleImageDownload(it, context) + } + DownloadedImage(getDefaultImageName(url), downloadedImage.first, downloadedImage.second) + } else { + // linked image ? + val value = headers.find { it.name.contains("content-type", true) }?.value + if (value != null) { + if (value.contains("text/html")) { + val document = fetch(url, context) { + htmlProcessorService.clean(it.entity.content) + } + if (log.isDebugEnabled) { + log.debug("Cleaning $url response", url) + } + val resolvedImage = resolve(url, document, context) + val downloadImage: Pair = + fetch(resolvedImage.second, context) { + handleImageDownload(it, context) + } + DownloadedImage(resolvedImage.first, downloadImage.first, downloadImage.second) + } else { + throw HostException("Unable to download $url, can't process content type $value") + } + } else { + throw HostException("Unexpected server response for $url, response have no content type") + } + } + return downloadedImage + } + + private fun handleImageDownload( + response: CloseableHttpResponse, + context: ImageDownloadContext + ): Pair { + val mimeType = getImageMimeType(response.allHeaders) + ?: throw HostException("Unsupported image type ${response.getFirstHeader("content-type")}") + + val tempImage = Files.createTempFile("vripper", "tmp") + return response.entity.content.use { inputStream -> + try { + FileOutputStream(tempImage.pathString).use { fos -> + val image = context.image + image.total = response.entity.contentLength + dataTransaction.update(image) + log.debug( + "Length is ${image.total}" + ) + log.debug( + "Starting data transfer" + ) + val buffer = ByteArray(READ_BUFFER_SIZE) + var read: Int + while (inputStream.read(buffer, 0, READ_BUFFER_SIZE) + .also { read = it } != -1 && !context.stopped + ) { + fos.write(buffer, 0, read) + with(image) { + current += read + } + downloadSpeedService.increase(read.toLong()) + dataTransaction.update(image) + } + Pair(tempImage, mimeType) + } + } finally { + EntityUtils.consumeQuietly(response.entity) + } + } + } + + fun isSupported(url: String): Boolean { + return url.contains(host) + } + + @Throws(HostException::class) + fun head(url: String, context: HttpClientContext): Array
{ + val client: HttpClient = httpService.client.build() + val httpGet = httpService.buildHttpHead(url, context) + log.debug(String.format("Requesting %s", url)) + return try { + (client.execute( + httpGet, + context + ) as CloseableHttpResponse).use { response -> + try { + if (response.statusLine.statusCode / 100 != 2) { + throw HostException( + String.format( + "Unexpected response code: %d", response.statusLine.statusCode + ) + ) + } + response.allHeaders + } finally { + EntityUtils.consumeQuietly(response.entity) + } + } + } catch (e: Exception) { + throw HostException(e) + } + } + + @Throws(HostException::class) + fun fetch( + url: String, + context: ImageDownloadContext, + transformer: (CloseableHttpResponse) -> T + ): T { + val client: HttpClient = httpService.client.build() + val httpGet = httpService.buildHttpGet(url, context.httpContext) + httpGet.addHeader("Referer", context.image.url) + log.debug(String.format("Requesting %s", url)) + return try { + (client.execute( + httpGet, + context.httpContext + ) as CloseableHttpResponse).use { response -> + if (response.statusLine.statusCode / 100 != 2) { + EntityUtils.consumeQuietly(response.entity) + throw DownloadException( + "Server returned code ${response.statusLine.statusCode}" + ) + } + try { + transformer(response) + } finally { + EntityUtils.consumeQuietly(response.entity) + } + } + } catch (e: Exception) { + throw HostException(e) + } + } + + private fun getImageMimeType(headers: Array
): ImageMimeType? { + + // first check if content type header exists + val value = headers.find { it.name.contains("content-type", true) }?.value + + // header found, check the type + return if (value != null) { + ImageMimeType.values().find { + value.contains(it.strValue, true) + } + } else { + null + } + } + + fun getDefaultImageName(imgUrl: String): String { + val imageTitle = imgUrl.substring(imgUrl.lastIndexOf('/') + 1) + log.debug(String.format("Extracting name from url %s: %s", imgUrl, imageTitle)) + return getFileNameWithoutExtension(imageTitle) + } + + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (javaClass != other?.javaClass) return false + + other as Host + + if (host != other.host) return false + + return true + } + + override fun hashCode(): Int { + return host.hashCode() + } +} + +data class DownloadedImage(val name: String, val path: Path, val type: ImageMimeType) + +enum class ImageMimeType(val strValue: String) { + IMAGE_BMP("image/bmp"), + IMAGE_GIF("image/gif"), + IMAGE_JPEG("image/jpeg"), + IMAGE_PNG("image/png"), + IMAGE_WEBP("image/webp"), +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageBamHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageBamHost.kt new file mode 100644 index 00000000..1a7e79f5 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageBamHost.kt @@ -0,0 +1,92 @@ +package me.mnlr.vripper.host + +import org.apache.http.impl.cookie.BasicClientCookie +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* +import java.sql.Date +import java.time.LocalDateTime +import java.time.ZoneId +import java.util.* + +@Service +class ImageBamHost( + private val htmlProcessorService: HtmlProcessorService, + private val xpathService: XpathService, + httpService: HTTPService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val doc = try { + log.debug(String.format("Looking for xpath expression %s in %s", CONTINUE_XPATH, url)) + if (xpathService.getAsNode(document, CONTINUE_XPATH) != null) { + val clientCookie = BasicClientCookie("nsfw_inter", "1") + clientCookie.domain = "www.imagebam.com" + clientCookie.path = "/" + clientCookie.expiryDate = + Date.from( + LocalDateTime.now().plusDays(3).atZone(ZoneId.systemDefault()).toInstant() + ) + context.httpContext.cookieStore.addCookie(clientCookie) + fetch(url, context) { + htmlProcessorService.clean(it.entity.content) + } + } else { + document + } + } catch (e: XpathException) { + throw HostException(e) + } + val imgNode: Node = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(doc, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + val imgTitle = Optional.ofNullable(imgNode.attributes.getNamedItem("alt")) + .map { e: Node -> e.textContent.trim { it <= ' ' } } + .orElse("") + val imgUrl = Optional.ofNullable(imgNode.attributes.getNamedItem("src")) + .map { e: Node -> e.textContent.trim { it <= ' ' } } + .orElse("") + var defaultName: String = UUID.randomUUID().toString() + val index = imgUrl.lastIndexOf('/') + if (index != -1 && index < imgUrl.length) { + defaultName = imgUrl.substring(imgUrl.lastIndexOf('/') + 1) + } + Pair((imgTitle.ifEmpty { defaultName })!!, imgUrl) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "imagebam.com" + private const val lookup = "imagebam.com" + private const val IMG_XPATH = "//img[contains(@class,'main-image')]" + private const val CONTINUE_XPATH = "//*[contains(text(), 'Continue')]" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageTwistHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageTwistHost.kt new file mode 100644 index 00000000..3ce874be --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageTwistHost.kt @@ -0,0 +1,61 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* +import java.util.* + +@Service +class ImageTwistHost( + private val xpathService: XpathService, + httpService: HTTPService, + htmlProcessorService: HtmlProcessorService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val imgNode: Node = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(document, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + val imgTitle = + Optional.ofNullable(imgNode.attributes.getNamedItem("alt")) + .map { obj: Node -> obj.textContent } + .map { obj: String -> obj.trim { it <= ' ' } }.orElse(null) + val imgUrl = imgNode.attributes.getNamedItem("src").textContent.trim { it <= ' ' } + Pair(imgTitle!!, imgUrl) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "imagetwist.com" + private const val lookup = "imagetwist.com" + private const val IMG_XPATH = "//img[contains(@class, 'img')]" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageVenueHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageVenueHost.kt new file mode 100644 index 00000000..0fb49d43 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageVenueHost.kt @@ -0,0 +1,80 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* + +@Service +class ImageVenueHost( + private val htmlProcessorService: HtmlProcessorService, + private val xpathService: XpathService, + httpService: HTTPService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val doc = try { + log.debug( + String.format( + "Looking for xpath expression %s in %s", + CONTINUE_BUTTON_XPATH, + url + ) + ) + if (xpathService.getAsNode(document, CONTINUE_BUTTON_XPATH) != null) { + // Button detected. No need to actually click it, just make the call again. + fetch(url, context) { + htmlProcessorService.clean(it.entity.content) + } + } else { + document + } + } catch (e: XpathException) { + throw HostException(e) + } + val imgNode: Node = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(doc, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + val imgTitle = imgNode.attributes.getNamedItem("alt").textContent.trim { it <= ' ' } + val imgUrl = imgNode.attributes.getNamedItem("src").textContent.trim { it <= ' ' } + Pair( + if (imgTitle.isEmpty()) imgUrl.substring(imgUrl.lastIndexOf('/') + 1) else imgTitle, + imgUrl + ) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "imagevenue.com" + private const val lookup = "imagevenue.com" + private const val CONTINUE_BUTTON_XPATH = "//a[@title='Continue to ImageVenue']" + private const val IMG_XPATH = "//a[@data-toggle='full']/img" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageZillaHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageZillaHost.kt new file mode 100644 index 00000000..29e03763 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImageZillaHost.kt @@ -0,0 +1,59 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* + +@Service +class ImageZillaHost( + private val xpathService: XpathService, + httpService: HTTPService, + htmlProcessorService: HtmlProcessorService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val titleNode = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(document, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + log.debug(String.format("Resolving name for %s", url)) + var title = titleNode.attributes.getNamedItem("title").textContent.trim() + titleNode.textContent.trim() + if (title.isEmpty()) { + title = getDefaultImageName(url) + } + return try { + Pair(title, url.replace("show", "images")) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "imagezilla.net" + private const val lookup = "imagezilla.net/show" + private const val IMG_XPATH = "//img[@id='photo']" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImgSpiceHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImgSpiceHost.kt new file mode 100644 index 00000000..9c353691 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImgSpiceHost.kt @@ -0,0 +1,59 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* + +@Service +class ImgSpiceHost( + private val xpathService: XpathService, + httpService: HTTPService, + htmlProcessorService: HtmlProcessorService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val imgNode: Node = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(document, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + val imgTitle = imgNode.attributes.getNamedItem("alt").textContent.trim { it <= ' ' } + val imgUrl = imgNode.attributes.getNamedItem("src").textContent.trim { it <= ' ' } + Pair( + imgTitle.ifEmpty { imgUrl.substring(imgUrl.lastIndexOf('/') + 1) }, imgUrl + ) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "imgspice.com" + private const val lookup = "imgspice.com" + private const val IMG_XPATH = "//img[@id='imgpreview']" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImgboxHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImgboxHost.kt new file mode 100644 index 00000000..09af3f13 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImgboxHost.kt @@ -0,0 +1,57 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* + +@Service +class ImgboxHost( + private val xpathService: XpathService, + httpService: HTTPService, + htmlProcessorService: HtmlProcessorService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val imgNode: Node = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(document, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + val imgTitle = imgNode.attributes.getNamedItem("title").textContent.trim { it <= ' ' } + val imgUrl = imgNode.attributes.getNamedItem("src").textContent.trim { it <= ' ' } + Pair(imgTitle, imgUrl) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "imgbox.com" + private const val lookup = "imgbox.com" + private const val IMG_XPATH = "//img[@id='img']" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImxHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImxHost.kt new file mode 100644 index 00000000..73d0764d --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ImxHost.kt @@ -0,0 +1,107 @@ +package me.mnlr.vripper.host + +import org.apache.http.NameValuePair +import org.apache.http.client.HttpClient +import org.apache.http.client.entity.UrlEncodedFormEntity +import org.apache.http.client.methods.CloseableHttpResponse +import org.apache.http.client.methods.HttpPost +import org.apache.http.message.BasicNameValuePair +import org.apache.http.util.EntityUtils +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.HtmlProcessorException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* +import java.io.IOException + +@Service +class ImxHost( + private val httpService: HTTPService, + private val htmlProcessorService: HtmlProcessorService, + private val xpathService: XpathService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + var value: String? = null + try { + log.debug("Looking for xpath expression $CONTINUE_BUTTON_XPATH in $url") + val contDiv = xpathService.getAsNode(document, CONTINUE_BUTTON_XPATH) + ?: throw HostException("$CONTINUE_BUTTON_XPATH cannot be found") + val node = contDiv.attributes.getNamedItem("value") + if (node != null) { + value = node.textContent + } + } catch (e: XpathException) { + throw HostException(e) + } + if (value == null) { + throw HostException("Failed to obtain value attribute from continue input") + } + log.debug("Click button found for $url") + val client: HttpClient = httpService.client.build() + val httpPost: HttpPost = httpService.buildHttpPost(url, context.httpContext) + val params: MutableList = ArrayList() + params.add(BasicNameValuePair("imgContinue", value)) + try { + httpPost.entity = UrlEncodedFormEntity(params) + } catch (e: Exception) { + throw HostException(e) + } + log.debug("Requesting $httpPost") + val doc = try { + (client.execute( + httpPost, context.httpContext + ) as CloseableHttpResponse).use { response -> + log.debug("Cleaning response for $httpPost") + try { + htmlProcessorService.clean(response.entity.content) + } finally { + EntityUtils.consumeQuietly(response.entity) + } + } + } catch (e: IOException) { + throw HostException(e) + } catch (e: HtmlProcessorException) { + throw HostException(e) + } + val imgNode: Node = try { + log.debug("Looking for xpath expression $IMG_XPATH in $url") + xpathService.getAsNode(doc, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + "Xpath $IMG_XPATH cannot be found in $url" + ) + return try { + log.debug("Resolving name and image url for $url") + val imgTitle = imgNode.attributes.getNamedItem("alt").textContent.trim { it <= ' ' } + val imgUrl = imgNode.attributes.getNamedItem("src").textContent.trim { it <= ' ' } + Pair( + imgTitle.ifEmpty { getDefaultImageName(imgUrl) }, imgUrl + ) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "imx.to" + private const val lookup = "imx.to" + private const val CONTINUE_BUTTON_XPATH = "//*[@name='imgContinue']" + private const val IMG_XPATH = "//img[@class='centred']" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PimpandhostHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PimpandhostHost.kt new file mode 100644 index 00000000..d7ebc6f2 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PimpandhostHost.kt @@ -0,0 +1,85 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* +import java.net.URI +import java.net.URISyntaxException + +@Service +class PimpandhostHost( + private val xpathService: XpathService, + private val htmlProcessorService: HtmlProcessorService, + httpService: HTTPService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val newUrl: String + try { + newUrl = appendUri(url.replace("http://", "https://"), "size=original") + } catch (e: Exception) { + throw HostException(e) + } + val doc = fetch(newUrl, context) { + htmlProcessorService.clean(it.entity.content) + } + val imgNode: Node = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, newUrl)) + xpathService.getAsNode(doc, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", newUrl)) + val imgTitle = imgNode.attributes.getNamedItem("alt").textContent.trim { it <= ' ' } + val imgUrl = + "https:" + imgNode.attributes.getNamedItem("src").textContent.trim { it <= ' ' } + Pair( + imgTitle.ifEmpty { imgUrl.substring(imgUrl.lastIndexOf('/') + 1) }, imgUrl + ) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + @Throws(URISyntaxException::class) + fun appendUri(uri: String, appendQuery: String): String { + val oldUri = URI(uri) + var newQuery = oldUri.query + if (newQuery == null) { + newQuery = appendQuery + } else { + newQuery += "&$appendQuery" + } + return URI( + oldUri.scheme, oldUri.authority, oldUri.path, newQuery, oldUri.fragment + ).toString() + } + + companion object { + private const val host = "pimpandhost.com" + private const val lookup = "pimpandhost.com" + private const val IMG_XPATH = "//img[contains(@class, 'original')]" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PixRouteHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PixRouteHost.kt new file mode 100644 index 00000000..56c3d43f --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PixRouteHost.kt @@ -0,0 +1,56 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* + +@Service +class PixRouteHost( + private val xpathService: XpathService, + httpService: HTTPService, + htmlProcessorService: HtmlProcessorService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val imgNode: Node = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(document, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + Pair(imgNode.attributes.getNamedItem("alt").textContent.trim { it <= ' ' }, + imgNode.attributes.getNamedItem("src").textContent.trim { it <= ' ' }) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "pixroute.com" + private const val lookup = "pixroute.com" + private const val IMG_XPATH = "//img[@id='imgpreview']" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PixhostHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PixhostHost.kt new file mode 100644 index 00000000..d0cb9e56 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PixhostHost.kt @@ -0,0 +1,57 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* + +@Service +class PixhostHost( + private val xpathService: XpathService, + httpService: HTTPService, + htmlProcessorService: HtmlProcessorService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val imgNode: Node = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(document, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + val imgTitle = imgNode.attributes.getNamedItem("alt").textContent.trim { it <= ' ' } + val imgUrl = imgNode.attributes.getNamedItem("src").textContent.trim { it <= ' ' } + Pair(imgTitle.substring(imgTitle.indexOf('_') + 1), imgUrl) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "pixhost.to" + private const val lookup = "pixhost.to" + private const val IMG_XPATH = "//img[@id='image']" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PixxxelsHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PixxxelsHost.kt new file mode 100644 index 00000000..92d6154a --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PixxxelsHost.kt @@ -0,0 +1,71 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* + +@Service +class PixxxelsHost( + private val xpathService: XpathService, + httpService: HTTPService, + htmlProcessorService: HtmlProcessorService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val imgNode = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(document, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + val titleNode = try { + log.debug(String.format("Looking for xpath expression %s in %s", TITLE_XPATH, url)) + xpathService.getAsNode(document, TITLE_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + TITLE_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + val imgTitle = titleNode.textContent.trim { it <= ' ' } + val imgUrl = imgNode.attributes.getNamedItem("href").textContent.trim { it <= ' ' } + Pair( + imgTitle.ifEmpty { imgUrl.substring(imgUrl.lastIndexOf('/') + 1) }, imgUrl + ) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "pixxxels.cc" + private const val lookup = "pixxxels.cc" + private const val IMG_XPATH = "//*[@id='download']" + private const val TITLE_XPATH = "//*[contains(@class,'imagename')]" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PostImgHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PostImgHost.kt new file mode 100644 index 00000000..5752ce3e --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/PostImgHost.kt @@ -0,0 +1,72 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* +import java.util.* + +@Service +class PostImgHost( + private val xpathService: XpathService, + httpService: HTTPService, + htmlProcessorService: HtmlProcessorService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val titleNode = try { + log.debug(String.format("Looking for xpath expression %s in %s", TITLE_XPATH, url)) + xpathService.getAsNode(document, TITLE_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + TITLE_XPATH, + url + ) + ) + val urlNode = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(document, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + val imgTitle = Optional.ofNullable(titleNode) + .map { node: Node -> node.textContent.trim { it <= ' ' } } + .orElseGet { getDefaultImageName(url) } + Pair(imgTitle, urlNode.attributes.getNamedItem("href").textContent.trim { it <= ' ' }) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val host = "postimg.cc" + private const val lookup = "postimg.cc" + private const val TITLE_XPATH = "//span[contains(@class,'imagename')]" + private const val IMG_XPATH = "//a[@id='download']" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/TurboImageHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/TurboImageHost.kt new file mode 100644 index 00000000..44757b76 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/TurboImageHost.kt @@ -0,0 +1,59 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* + +@Service +class TurboImageHost( + private val xpathService: XpathService, + httpService: HTTPService, + htmlProcessorService: HtmlProcessorService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String + get() = Companion.host + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + var title: String? + title = try { + log.debug(String.format("Looking for xpath expression %s in %s", TITLE_XPATH, url)) + val titleNode: Node? = xpathService.getAsNode(document, TITLE_XPATH) + log.debug(String.format("Resolving name for %s", url)) + titleNode?.textContent?.trim { it <= ' ' } + } catch (e: XpathException) { + throw HostException(e) + } + if (title.isNullOrEmpty()) { + title = getDefaultImageName(url) + } + val urlNode: Node = xpathService.getAsNode(document, IMG_XPATH) + ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return Pair(title, urlNode.attributes.getNamedItem("src").textContent.trim { it <= ' ' }) + } + + companion object { + private const val host = "turboimagehost.com" + private const val lookup = "turboimagehost.com" + private const val TITLE_XPATH = "//div[contains(@class,'titleFullS')]/h1" + private const val IMG_XPATH = "//img[@id='imageid']" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ViprImHost.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ViprImHost.kt new file mode 100644 index 00000000..12e6d588 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/host/ViprImHost.kt @@ -0,0 +1,58 @@ +package me.mnlr.vripper.host + +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.ImageDownloadContext +import me.mnlr.vripper.exception.HostException +import me.mnlr.vripper.exception.XpathException +import me.mnlr.vripper.services.* +import java.util.* + +@Service +class ViprImHost( + private val xpathService: XpathService, + httpService: HTTPService, + htmlProcessorService: HtmlProcessorService, + dataTransaction: DataTransaction, + downloadSpeedService: DownloadSpeedService, +) : Host(httpService, htmlProcessorService, dataTransaction, downloadSpeedService) { + private val log by LoggerDelegate() + override val host: String = "vipr.im" + + @Throws(HostException::class) + override fun resolve( + url: String, + document: Document, + context: ImageDownloadContext + ): Pair { + val imgNode: Node = try { + log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)) + xpathService.getAsNode(document, IMG_XPATH) + } catch (e: XpathException) { + throw HostException(e) + } ?: throw HostException( + String.format( + "Xpath '%s' cannot be found in '%s'", + IMG_XPATH, + url + ) + ) + return try { + log.debug(String.format("Resolving name and image url for %s", url)) + val imgTitle = + Optional.ofNullable(imgNode.attributes.getNamedItem("alt")) + .map { obj: Node -> obj.textContent } + .map { obj: String -> obj.trim() }.orElse(null) + val imgUrl = imgNode.attributes.getNamedItem("src").textContent.trim() + Pair(imgTitle!!, imgUrl) + } catch (e: Exception) { + throw HostException("Unexpected error occurred", e) + } + } + + companion object { + private const val IMG_XPATH = "//img[contains(@class, 'img')]" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/model/DownloadSpeed.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/DownloadSpeed.kt new file mode 100644 index 00000000..30ac3398 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/DownloadSpeed.kt @@ -0,0 +1,19 @@ +package me.mnlr.vripper.model + +data class DownloadSpeed(val speed: String) { + + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (javaClass != other?.javaClass) return false + + other as DownloadSpeed + + if (speed != other.speed) return false + + return true + } + + override fun hashCode(): Int { + return speed.hashCode() + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/model/GlobalState.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/GlobalState.kt new file mode 100644 index 00000000..9941459c --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/GlobalState.kt @@ -0,0 +1,33 @@ +package me.mnlr.vripper.model + +data class GlobalState( + val running: Int, + val remaining: Int, + val error: Int, + val loggedUser: String, + val downloadSpeed: String +) { + override fun equals(other: Any?): Boolean { + if (this === other) return true + if (javaClass != other?.javaClass) return false + + other as GlobalState + + if (running != other.running) return false + if (remaining != other.remaining) return false + if (error != other.error) return false + if (loggedUser != other.loggedUser) return false + if (downloadSpeed != other.downloadSpeed) return false + + return true + } + + override fun hashCode(): Int { + var result = running + result = 31 * result + remaining + result = 31 * result + error + result = 31 * result + loggedUser.hashCode() + result = 31 * result + downloadSpeed.hashCode() + return result + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/model/ImageItem.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/ImageItem.kt new file mode 100644 index 00000000..3bcca2ea --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/ImageItem.kt @@ -0,0 +1,5 @@ +package me.mnlr.vripper.model + +import me.mnlr.vripper.host.Host + +data class ImageItem(val mainLink: String, val thumbLink: String, val host: Host) \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/model/LoggedUser.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/LoggedUser.kt new file mode 100644 index 00000000..fa824e60 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/LoggedUser.kt @@ -0,0 +1,3 @@ +package me.mnlr.vripper.model + +data class LoggedUser(val user: String) \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/model/PostItem.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/PostItem.kt new file mode 100644 index 00000000..86353ca2 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/PostItem.kt @@ -0,0 +1,17 @@ +package me.mnlr.vripper.model + +import me.mnlr.vripper.host.Host + +data class PostItem( + val threadId: String, + val threadTitle: String, + val postId: String, + val number: Int, + val title: String, + val imageCount: Int, + val url: String, + val hosts: Map, + val securityToken: String, + val forum: String, + val imageItemList: List +) \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/model/Settings.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/Settings.kt new file mode 100644 index 00000000..244477c4 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/Settings.kt @@ -0,0 +1,35 @@ +package me.mnlr.vripper.model + +class Settings { + var desktopClipboard: Boolean = false + var maxEventLog: Int = 1_000 + var connectionSettings: ConnectionSettings = ConnectionSettings() + var downloadSettings: DownloadSettings = DownloadSettings() + var viperSettings: ViperSettings = ViperSettings() +} + +data class ViperSettings( + var login: Boolean = false, + var username: String = "", + var password: String = "", + var thanks: Boolean = false, + var host: String = "https://vipergirls.to", +) + +data class DownloadSettings( + var downloadPath: String = System.getProperty("user.home"), + var autoStart: Boolean = true, + var autoQueueThreshold: Int = 1, + var forceOrder: Boolean = false, + var forumSubfolder: Boolean = false, + var threadSubLocation: Boolean = false, + var clearCompleted: Boolean = false, + var appendPostId: Boolean = false +) + +data class ConnectionSettings( + var maxThreads: Int = 4, + var maxTotalThreads: Int = 0, + var timeout: Int = 30, + var maxAttempts: Int = 3, +) \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/model/ThreadInput.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/ThreadInput.kt new file mode 100644 index 00000000..af23af2c --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/ThreadInput.kt @@ -0,0 +1,3 @@ +package me.mnlr.vripper.model + +data class ThreadInput(val link: String, val threadId: String) \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/model/ThreadItem.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/ThreadItem.kt new file mode 100644 index 00000000..014e68e8 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/model/ThreadItem.kt @@ -0,0 +1,9 @@ +package me.mnlr.vripper.model + +data class ThreadItem( + val threadId: String, + val title: String, + val securityToken: String, + val forum: String, + val postItemList: List +) diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/parser/ThreadLookupAPIParser.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/parser/ThreadLookupAPIParser.kt new file mode 100644 index 00000000..54bb2626 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/parser/ThreadLookupAPIParser.kt @@ -0,0 +1,79 @@ +package me.mnlr.vripper.parser + +import net.jodah.failsafe.Failsafe +import net.jodah.failsafe.function.CheckedSupplier +import org.apache.http.client.HttpClient +import org.apache.http.client.methods.CloseableHttpResponse +import org.apache.http.client.methods.HttpGet +import org.apache.http.client.protocol.HttpClientContext +import org.apache.http.client.utils.URIBuilder +import org.apache.http.util.EntityUtils +import me.mnlr.vripper.SpringContext +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.exception.DownloadException +import me.mnlr.vripper.exception.PostParseException +import me.mnlr.vripper.model.ThreadItem +import me.mnlr.vripper.services.HTTPService +import me.mnlr.vripper.services.RetryPolicyService +import me.mnlr.vripper.services.SettingsService +import me.mnlr.vripper.services.VGAuthService +import java.io.BufferedInputStream +import java.net.URISyntaxException +import java.util.* +import javax.xml.parsers.SAXParserFactory + +class ThreadLookupAPIParser(private val threadId: String) { + private val log by LoggerDelegate() + private val cm: HTTPService = SpringContext.getBean(HTTPService::class.java) + private val retryPolicyService: RetryPolicyService = + SpringContext.getBean(RetryPolicyService::class.java) + private val vgAuthService: VGAuthService = SpringContext.getBean(VGAuthService::class.java) + private val settingsService: SettingsService = + SpringContext.getBean(SettingsService::class.java) + + @Throws(PostParseException::class) + fun parse(): ThreadItem { + log.debug("Parsing thread $threadId") + val httpGet: HttpGet = try { + val uriBuilder = URIBuilder(settingsService.settings.viperSettings.host + "/vr.php") + uriBuilder.setParameter("t", threadId) + cm.buildHttpGet(uriBuilder.build(), HttpClientContext.create()) + } catch (e: URISyntaxException) { + throw PostParseException(e) + } + val threadLookupAPIResponseHandler = ThreadLookupAPIResponseHandler() + log.debug("Requesting $httpGet") + return try { + Failsafe.with(retryPolicyService.buildGenericRetryPolicy()).onFailure { + log.error( + "parsing failed for thread $threadId", + it.failure + ) + }.get(CheckedSupplier { + val connection: HttpClient = cm.client.build() + (connection.execute( + httpGet, vgAuthService.context + ) as CloseableHttpResponse).use { response -> + try { + if (response.statusLine.statusCode / 100 != 2) { + throw DownloadException("Unexpected response code '${response.statusLine.statusCode}' for $httpGet") + } + factory.newSAXParser().parse( + BufferedInputStream(response.entity.content), + threadLookupAPIResponseHandler + ) + threadLookupAPIResponseHandler.result + } finally { + EntityUtils.consumeQuietly(response.entity) + } + } + }) + } catch (e: Exception) { + throw PostParseException(e) + } + } + + companion object { + private val factory = SAXParserFactory.newInstance() + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/parser/ThreadLookupAPIResponseHandler.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/parser/ThreadLookupAPIResponseHandler.kt new file mode 100644 index 00000000..2625b4d1 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/parser/ThreadLookupAPIResponseHandler.kt @@ -0,0 +1,104 @@ +package me.mnlr.vripper.parser + +import org.xml.sax.Attributes +import org.xml.sax.helpers.DefaultHandler +import me.mnlr.vripper.SpringContext +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.host.Host +import me.mnlr.vripper.model.ImageItem +import me.mnlr.vripper.model.PostItem +import me.mnlr.vripper.model.ThreadItem +import me.mnlr.vripper.services.SettingsService + +class ThreadLookupAPIResponseHandler : DefaultHandler() { + private val log by LoggerDelegate() + private val supportedHosts: Collection = + SpringContext.getBeansOfType(Host::class.java).values + private val settingsService: SettingsService = + SpringContext.getBean(SettingsService::class.java) + private var error: String = "" + private val hostMap: MutableMap = mutableMapOf() + private val postItemList: MutableList = mutableListOf() + private val imageItemList: MutableList = mutableListOf() + private lateinit var threadItem: ThreadItem + private lateinit var threadId: String + private lateinit var threadTitle: String + private lateinit var forum: String + private lateinit var securityToken: String + private lateinit var postId: String + private lateinit var postTitle: String + private var postCounter: Int = 0 + + val result: ThreadItem + get() = threadItem + + override fun startDocument() {} + override fun startElement( + uri: String, localName: String, qName: String, attributes: Attributes + ) { + when (qName.lowercase()) { + "error" -> error = attributes.getValue("details") + "thread" -> { + threadId = attributes.getValue("id")?.trim() ?: "" + threadTitle = attributes.getValue("title")?.trim() ?: "" + } + + "forum" -> forum = attributes.getValue("title")?.trim() ?: "" + "user" -> securityToken = attributes.getValue("hash")?.trim() ?: "" + "post" -> { + postId = attributes.getValue("id")?.trim() ?: "" + postCounter = attributes.getValue("number")?.trim()?.toInt() ?: 0 + val title = attributes.getValue("title")?.trim() ?: "" + postTitle = title.ifBlank { threadTitle } + } + + "image" -> { + val mainLink = attributes.getValue("main_url")?.trim() ?: "" + val thumbLink = attributes.getValue("thumb_url")?.trim() ?: "" + val type = attributes.getValue("type")?.trim() ?: "" + if(type == "linked") { + try { + val host = supportedHosts.first { + it.isSupported(mainLink) + }.let { host -> + hostMap.computeIfAbsent(host) { 0 } + hostMap[host] = hostMap[host]!! + 1 + host + } + imageItemList.add(ImageItem(mainLink, thumbLink, host)) + } catch (e: Exception) { + log.warn("Unsupported link: $mainLink") + } + } + } + } + } + + override fun endElement(uri: String, localName: String, qName: String) { + if ("post".equals(qName, true)) { + if (imageItemList.isNotEmpty()) { + postItemList.add( + PostItem( + threadId, + threadTitle, + postId, + postCounter, + postTitle, + imageItemList.size, + "${settingsService.settings.viperSettings.host}/threads/?p=$postId&viewfull=1#post$postId", + hostMap.toMap(), + securityToken, + forum, + imageItemList.toList() + ) + ) + } + imageItemList.clear() + hostMap.clear() + } + } + + override fun endDocument() { + threadItem = ThreadItem(threadId, threadTitle, securityToken, forum, postItemList.toList()) + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/ImageRepository.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/ImageRepository.kt new file mode 100644 index 00000000..1c3a7c03 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/ImageRepository.kt @@ -0,0 +1,16 @@ +package me.mnlr.vripper.repositories + +import me.mnlr.vripper.entities.ImageDownloadState +import java.util.* + +interface ImageRepository { + fun save(imageDownloadState: ImageDownloadState): ImageDownloadState + fun deleteAllByPostId(postId: String) + fun findByPostId(postId: String): List + fun countError(): Int + fun findByPostIdAndIsNotCompleted(postId: String): List + fun stopByPostIdAndIsNotCompleted(postId: String): Int + fun findByPostIdAndIsError(postId: String): List + fun findById(id: Long): Optional + fun update(imageDownloadState: ImageDownloadState) +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/LogEventRepository.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/LogEventRepository.kt new file mode 100644 index 00000000..24db701f --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/LogEventRepository.kt @@ -0,0 +1,13 @@ +package me.mnlr.vripper.repositories + +import me.mnlr.vripper.entities.LogEvent +import java.util.* + +interface LogEventRepository { + fun save(logEvent: LogEvent): LogEvent + fun update(logEvent: LogEvent): LogEvent + fun findAll(): List + fun findById(id: Long): Optional + fun delete(id: Long) + fun deleteAll() +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/MetadataRepository.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/MetadataRepository.kt new file mode 100644 index 00000000..e7134f8d --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/MetadataRepository.kt @@ -0,0 +1,10 @@ +package me.mnlr.vripper.repositories + +import me.mnlr.vripper.entities.Metadata +import java.util.* + +interface MetadataRepository { + fun save(metadata: Metadata): Metadata + fun findByPostId(postId: String): Optional + fun deleteByPostId(postId: String): Int +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/PostDownloadStateRepository.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/PostDownloadStateRepository.kt new file mode 100644 index 00000000..2ca3688f --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/PostDownloadStateRepository.kt @@ -0,0 +1,16 @@ +package me.mnlr.vripper.repositories + +import me.mnlr.vripper.entities.PostDownloadState +import java.util.* + +interface PostDownloadStateRepository { + fun save(postDownloadState: PostDownloadState): PostDownloadState + fun findByPostId(postId: String): Optional + fun findById(id: Long): Optional + fun findCompleted(): List + fun findAll(): List + fun existByPostId(postId: String): Boolean + fun setDownloadingToStopped(): Int + fun deleteByPostId(postId: String): Int + fun update(postDownloadState: PostDownloadState) +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/ThreadRepository.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/ThreadRepository.kt new file mode 100644 index 00000000..03efb3d1 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/ThreadRepository.kt @@ -0,0 +1,13 @@ +package me.mnlr.vripper.repositories + +import me.mnlr.vripper.entities.Thread +import java.util.* + +interface ThreadRepository { + fun save(thread: Thread): Thread + fun findByThreadId(threadId: String): Optional + fun findAll(): List + fun findById(id: Long): Optional + fun deleteByThreadId(threadId: String): Int + fun deleteAll() +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/ImageRepositoryImpl.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/ImageRepositoryImpl.kt new file mode 100644 index 00000000..8acdeb0d --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/ImageRepositoryImpl.kt @@ -0,0 +1,125 @@ +package me.mnlr.vripper.repositories.impl + +import org.springframework.jdbc.core.JdbcTemplate +import org.springframework.jdbc.core.RowMapper +import org.springframework.stereotype.Service +import me.mnlr.vripper.SpringContext +import me.mnlr.vripper.entities.ImageDownloadState +import me.mnlr.vripper.entities.domain.Status +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.host.Host +import me.mnlr.vripper.repositories.ImageRepository +import java.sql.ResultSet +import java.sql.SQLException +import java.util.* + +@Service +class ImageRepositoryImpl( + private val jdbcTemplate: JdbcTemplate, private val eventBus: EventBus +) : ImageRepository { + @Synchronized + private fun nextId(): Long { + return jdbcTemplate.queryForObject("CALL NEXT VALUE FOR SEQ_IMAGE", Long::class.java)!! + } + + override fun save(imageDownloadState: ImageDownloadState): ImageDownloadState { + val id = nextId() + jdbcTemplate.update( + "INSERT INTO IMAGE (ID, CURRENT, HOST, INDEX, POST_ID, STATUS, TOTAL, URL, POST_ID_REF) VALUES (?,?,?,?,?,?,?,?,?)", + id, + imageDownloadState.current, + imageDownloadState.host.host, + imageDownloadState.index, + imageDownloadState.postId, + imageDownloadState.status.name, + imageDownloadState.total, + imageDownloadState.url, + imageDownloadState.postIdRef + ) + imageDownloadState.id = id + eventBus.publishEvent(Event(Event.Kind.IMAGE_UPDATE, id)) + return imageDownloadState + } + + override fun deleteAllByPostId(postId: String) { + jdbcTemplate.update("DELETE FROM IMAGE WHERE POST_ID = ?", postId) + } + + override fun findByPostId(postId: String): List { + return jdbcTemplate.query( + "SELECT * FROM IMAGE WHERE POST_ID = ?", ImageRowMapper(), postId + ) + } + + override fun countError(): Int { + return jdbcTemplate.queryForObject( + "SELECT COUNT(*) FROM IMAGE AS image WHERE image.STATUS = ?", Int::class.java, Status.ERROR.name + ) + } + + override fun findByPostIdAndIsNotCompleted(postId: String): List { + return jdbcTemplate.query( + "SELECT * FROM IMAGE AS image WHERE image.POST_ID = ? AND image.STATUS <> ?", + ImageRowMapper(), + postId, + Status.FINISHED.name + ) + } + + override fun stopByPostIdAndIsNotCompleted(postId: String): Int { + return jdbcTemplate.update( + "UPDATE IMAGE AS image SET image.STATUS = ? WHERE image.POST_ID = ? AND image.STATUS <> ?", + Status.STOPPED.name, + postId, + Status.FINISHED.name + ) + } + + override fun findByPostIdAndIsError(postId: String): List { + return jdbcTemplate.query( + "SELECT * FROM IMAGE AS image WHERE image.POST_ID = ? AND image.STATUS = ?", + ImageRowMapper(), + postId, + Status.ERROR.name + ) + } + + override fun findById(id: Long): Optional { + val images = jdbcTemplate.query( + "SELECT * FROM IMAGE AS image WHERE image.ID = ?", ImageRowMapper(), id + ) + return if (images.isEmpty()) { + Optional.empty() + } else { + Optional.of(images[0]) + } + } + + override fun update(imageDownloadState: ImageDownloadState) { + jdbcTemplate.update( + "UPDATE IMAGE AS image SET image.STATUS = ?, image.CURRENT = ?, image.TOTAL = ? WHERE image.ID = ?", + imageDownloadState.status.name, + imageDownloadState.current, + imageDownloadState.total, + imageDownloadState.id + ) + eventBus.publishEvent(Event(Event.Kind.IMAGE_UPDATE, imageDownloadState.id)) + } +} + +internal class ImageRowMapper : RowMapper { + @Throws(SQLException::class) + override fun mapRow(rs: ResultSet, rowNum: Int): ImageDownloadState { + val id = rs.getLong("ID") + val postId = rs.getString("POST_ID") + val url = rs.getString("URL") + val host = SpringContext.getBeansOfType(Host::class.java).values.filter { it.host == rs.getString("HOST") }[0] + val index = rs.getInt("INDEX") + val current = rs.getLong("CURRENT") + val total = rs.getLong("TOTAL") + val status = Status.valueOf(rs.getString("STATUS")) + val postIdRef = rs.getLong("POST_ID_REF") + return ImageDownloadState(id, postId, url, host, index, postIdRef, total, current, status) + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/LogEventRepositoryImpl.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/LogEventRepositoryImpl.kt new file mode 100644 index 00000000..a6193993 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/LogEventRepositoryImpl.kt @@ -0,0 +1,107 @@ +package me.mnlr.vripper.repositories.impl + +import org.springframework.jdbc.core.JdbcTemplate +import org.springframework.jdbc.core.RowMapper +import org.springframework.stereotype.Service +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.entities.LogEvent +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.repositories.LogEventRepository +import me.mnlr.vripper.services.SettingsService +import java.sql.ResultSet +import java.sql.SQLException +import java.time.LocalDateTime +import java.time.format.DateTimeFormatter +import java.util.* + +@Service +class LogEventRepositoryImpl( + private val jdbcTemplate: JdbcTemplate, + private val settingsService: SettingsService, + private val eventBus: EventBus +) : LogEventRepository { + + private val log by LoggerDelegate() + + @Synchronized + private fun nextId(): Long { + return jdbcTemplate.queryForObject("CALL NEXT VALUE FOR SEQ_EVENT", Long::class.java)!! + } + + @Synchronized + override fun save(logEvent: LogEvent): LogEvent { + val maxRecords = settingsService.settings.maxEventLog - 1 + val count = jdbcTemplate.queryForObject("SELECT COUNT(*) FROM EVENT", Long::class.java) ?: 0 + if (count > maxRecords) { + val idList = jdbcTemplate.queryForList( + "SELECT ID FROM EVENT ORDER BY TIME ASC LIMIT ?", + Long::class.java, + count - maxRecords + ) + idList.forEach { delete(it) } + } + val id = nextId() + jdbcTemplate.update( + "INSERT INTO EVENT (ID, TYPE, STATUS, TIME, MESSAGE) VALUES (?,?,?,?,?)", + id, + logEvent.type.name, + logEvent.status.name, + logEvent.time.format(DateTimeFormatter.ISO_LOCAL_DATE_TIME), + logEvent.message + ) + eventBus.publishEvent(Event(Event.Kind.LOG_EVENT_UPDATE, id)) + return logEvent.copy(id = id) + } + + override fun update(logEvent: LogEvent): LogEvent { + if (logEvent.id == null) { + log.warn("Cannot update entity with null id") + return logEvent + } + jdbcTemplate.update( + "UPDATE EVENT SET STATUS = ?, MESSAGE = ? WHERE ID = ?", + logEvent.status.name, + logEvent.message, + logEvent.id + ) + eventBus.publishEvent(Event(Event.Kind.LOG_EVENT_UPDATE, logEvent.id)) + return logEvent + } + + override fun findById(id: Long): Optional { + val logEvents = + jdbcTemplate.query("SELECT * FROM EVENT WHERE ID = ?", LogEventRowMapper(), id) + return if (logEvents.isEmpty()) { + Optional.empty() + } else { + Optional.of(logEvents[0]) + } + } + + override fun findAll(): List { + return jdbcTemplate.query("SELECT * FROM EVENT", LogEventRowMapper()) + } + + override fun delete(id: Long) { + jdbcTemplate.update("DELETE FROM EVENT WHERE ID = ?", id) + eventBus.publishEvent(Event(Event.Kind.LOG_EVENT_REMOVE, id)) + } + + override fun deleteAll() { + jdbcTemplate.update("DELETE FROM EVENT") + } +} + +internal class LogEventRowMapper : RowMapper { + @Throws(SQLException::class) + override fun mapRow(rs: ResultSet, rowNum: Int): LogEvent { + val id = rs.getLong("ID") + val type = LogEvent.Type.valueOf(rs.getString("TYPE")) + val status = LogEvent.Status.valueOf(rs.getString("STATUS")) + val time = + LocalDateTime.parse(rs.getString("TIME"), DateTimeFormatter.ISO_LOCAL_DATE_TIME) + val message = rs.getString("MESSAGE") + return LogEvent(id, type, status, time, message) + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/MetadataRepositoryImpl.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/MetadataRepositoryImpl.kt new file mode 100644 index 00000000..a0ea1f42 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/MetadataRepositoryImpl.kt @@ -0,0 +1,65 @@ +package me.mnlr.vripper.repositories.impl + +import org.springframework.jdbc.core.JdbcTemplate +import org.springframework.jdbc.core.RowMapper +import org.springframework.stereotype.Service +import me.mnlr.vripper.entities.Metadata +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.repositories.* +import java.sql.ResultSet +import java.sql.SQLException +import java.util.* + +@Service +class MetadataRepositoryImpl( + private val jdbcTemplate: JdbcTemplate, + private val eventBus: EventBus +) : MetadataRepository { + override fun save(metadata: Metadata): Metadata { + jdbcTemplate.update( + "INSERT INTO METADATA (POST_ID_REF, POST_ID, POSTED_BY, RESOLVED_NAMES) VALUES (?,?,?,?)", + metadata.postIdRef, + metadata.postId, + metadata.postedBy, + java.lang.String.join("%sep%", metadata.resolvedNames) + ) + eventBus.publishEvent(Event(Event.Kind.METADATA_UPDATE, metadata.postIdRef)) + return metadata + } + + override fun findByPostId(postId: String): Optional { + val metadata = jdbcTemplate.query( + "SELECT metadata.* FROM METADATA AS metadata WHERE metadata.POST_ID = ?", + MetadataRowMapper(), + postId + ) + return if (metadata.isEmpty()) { + Optional.empty() + } else { + Optional.of(metadata[0]) + } + } + + override fun deleteByPostId(postId: String): Int { + return jdbcTemplate.update( + "DELETE FROM METADATA AS metadata WHERE metadata.POST_ID = ?", postId + ) + } +} + +internal class MetadataRowMapper : RowMapper { + @Throws(SQLException::class) + override fun mapRow(rs: ResultSet, rowNum: Int): Metadata { + val metadata = Metadata() + metadata.postIdRef = rs.getLong("POST_ID_REF") + metadata.postId = rs.getString("POST_ID") + metadata.postedBy = rs.getString("POSTED_BY") + val resolvedNames = rs.getString("RESOLVED_NAMES") + if (resolvedNames != null && resolvedNames.isNotBlank()) { + metadata.resolvedNames = + resolvedNames.split("%sep%").dropLastWhile { it.isEmpty() } + } + return metadata + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/PostDownloadStateRepositoryImpl.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/PostDownloadStateRepositoryImpl.kt new file mode 100644 index 00000000..44b36050 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/PostDownloadStateRepositoryImpl.kt @@ -0,0 +1,163 @@ +package me.mnlr.vripper.repositories.impl + +import org.springframework.jdbc.core.JdbcTemplate +import org.springframework.jdbc.core.RowMapper +import org.springframework.stereotype.Service +import me.mnlr.vripper.entities.PostDownloadState +import me.mnlr.vripper.entities.domain.Status +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.repositories.PostDownloadStateRepository +import java.sql.ResultSet +import java.sql.SQLException +import java.sql.Timestamp +import java.util.* + +@Service +class PostDownloadStateRepositoryImpl( + private val jdbcTemplate: JdbcTemplate, private val eventBus: EventBus +) : PostDownloadStateRepository { + @Synchronized + private fun nextId(): Long { + return jdbcTemplate.queryForObject("CALL NEXT VALUE FOR SEQ_POST", Long::class.java)!! + } + + override fun save(postDownloadState: PostDownloadState): PostDownloadState { + val id = nextId() + jdbcTemplate.update( + "INSERT INTO POST (ID, DONE, HOSTS, POST_FOLDER_NAME, POST_ID, STATUS, THREAD_ID, POST_TITLE, THREAD_TITLE, FORUM, TOTAL, URL, TOKEN, ADDED_ON, RANK) VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)", + id, + postDownloadState.done, + java.lang.String.join(";", postDownloadState.hosts), + postDownloadState.downloadDirectory, + postDownloadState.postId, + postDownloadState.status.name, + postDownloadState.threadId, + postDownloadState.postTitle, + postDownloadState.threadTitle, + postDownloadState.forum, + postDownloadState.total, + postDownloadState.url, + postDownloadState.token, + Timestamp.valueOf(postDownloadState.addedOn), + postDownloadState.rank + ) + eventBus.publishEvent(Event(Event.Kind.POST_UPDATE, id)) + return postDownloadState.copy(id = id) + } + + override fun findByPostId(postId: String): Optional { + val post = jdbcTemplate.query( + "SELECT metadata.*,post.* FROM METADATA metadata FULL JOIN POST post ON metadata.POST_ID_REF = post.ID WHERE post.POST_ID = ?", + PostRowMapper(), + postId + ) + return if (post.isEmpty()) { + Optional.empty() + } else { + Optional.of(post[0]) + } + } + + override fun findCompleted(): List { + return jdbcTemplate.query( + "SELECT POST_ID FROM POST AS post WHERE status = ? AND done >= total", + { rs: ResultSet, _: Int -> rs.getString("POST_ID") }, + Status.FINISHED.name + ) + } + + override fun findById(id: Long): Optional { + val post = jdbcTemplate.query( + "SELECT metadata.*,post.* FROM METADATA metadata FULL JOIN POST post ON metadata.POST_ID_REF = post.ID WHERE post.ID = ?", + PostRowMapper(), + id + ) + return if (post.isEmpty()) { + Optional.empty() + } else { + Optional.of(post[0]) + } + } + + override fun findAll(): List { + return jdbcTemplate.query( + "SELECT metadata.*,post.* FROM METADATA metadata FULL JOIN POST post ON metadata.POST_ID_REF = post.ID", + PostRowMapper() + ) + } + + override fun existByPostId(postId: String): Boolean { + val count = jdbcTemplate.queryForObject( + "SELECT COUNT(*) FROM POST AS post WHERE post.POST_ID = ?", Int::class.java, postId + ) + return count > 0 + } + + override fun setDownloadingToStopped(): Int { + return jdbcTemplate.update( + "UPDATE POST AS post SET post.STATUS = ? WHERE post.STATUS = ? OR post.STATUS = ?", + Status.STOPPED.name, + Status.DOWNLOADING.name, + Status.PENDING.name + ) + } + + override fun deleteByPostId(postId: String): Int { + val mutationCount = jdbcTemplate.update("DELETE FROM POST AS post WHERE post.POST_ID = ?", postId) + eventBus.publishEvent(Event(Event.Kind.POST_REMOVE, postId)) + return mutationCount + } + + override fun update(postDownloadState: PostDownloadState) { + jdbcTemplate.update( + "UPDATE POST AS post SET post.STATUS = ?, post.DONE = ?, post.RANK = ? WHERE post.ID = ?", + postDownloadState.status.name, + postDownloadState.done, + postDownloadState.rank, + postDownloadState.id + ) + eventBus.publishEvent(Event(Event.Kind.POST_UPDATE, postDownloadState.id)) + } +} + +internal class PostRowMapper : RowMapper { + @Throws(SQLException::class) + override fun mapRow(rs: ResultSet, rowNum: Int): PostDownloadState { + val id = rs.getLong("post.ID") + val status = Status.valueOf(rs.getString("post.STATUS")) + val postId = rs.getString("post.POST_ID") + val threadId = rs.getString("post.THREAD_ID") + val postTitle = rs.getString("post.POST_TITLE") + val threadTitle = rs.getString("post.THREAD_TITLE") + val forum = rs.getString("post.FORUM") + val url = rs.getString("post.URL") + val token = rs.getString("post.TOKEN") + val done = rs.getInt("post.DONE") + val total = rs.getInt("post.TOTAL") + val hosts = rs.getString("post.HOSTS").split(DELIMITER).dropLastWhile { it.isEmpty() }.toSet() + val downloadDirectory = rs.getString("post.POST_FOLDER_NAME") + val addedOn = rs.getTimestamp("post.ADDED_ON").toLocalDateTime() + val rank = rs.getInt("post.RANK") +// val metadataId = rs.getLong("metadata.POST_ID_REF") +// if (!rs.wasNull()) { +// val metadata = Metadata() +// metadata.postIdRef = metadataId +// metadata.postId = rs.getString("metadata.POST_ID") +// val resolvedNames = rs.getString("metadata.RESOLVED_NAMES") +// if (resolvedNames != null && resolvedNames.isNotBlank()) { +// metadata.resolvedNames = +// resolvedNames.split("%sep%").dropLastWhile { it.isEmpty() } +// } +// metadata.postedBy = rs.getString("metadata.POSTED_BY") +// postDownloadState.metadata = metadata +// } + return PostDownloadState( + id, postTitle, threadTitle, forum, url, token, postId, threadId, total, hosts, downloadDirectory, addedOn, status, done, rank + ) + } + + companion object { + private const val DELIMITER = ";" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/ThreadRepositoryImpl.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/ThreadRepositoryImpl.kt new file mode 100644 index 00000000..80160e01 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/repositories/impl/ThreadRepositoryImpl.kt @@ -0,0 +1,87 @@ +package me.mnlr.vripper.repositories.impl + +import org.springframework.beans.factory.annotation.Autowired +import org.springframework.jdbc.core.JdbcTemplate +import org.springframework.jdbc.core.RowMapper +import org.springframework.stereotype.Service +import me.mnlr.vripper.entities.Thread +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.repositories.ThreadRepository +import java.sql.ResultSet +import java.sql.SQLException +import java.util.* + +@Service +class ThreadRepositoryImpl @Autowired constructor( + private val jdbcTemplate: JdbcTemplate, + private val eventBus: EventBus +) : ThreadRepository { + @Synchronized + private fun nextId(): Long { + return jdbcTemplate.queryForObject("CALL NEXT VALUE FOR SEQ_THREAD", Long::class.java)!! + } + + override fun save(thread: Thread): Thread { + val id = nextId() + jdbcTemplate.update( + "INSERT INTO THREAD (ID, TOTAL, LINK, THREAD_ID) values (?,?,?,?)", + id, + thread.total, + thread.link, + thread.threadId + ) + eventBus.publishEvent(Event(Event.Kind.THREAD_UPDATE, id)) + return thread.copy(id = id) + } + + override fun findByThreadId(threadId: String): Optional { + val threadList = jdbcTemplate.query( + "SELECT * FROM THREAD AS thread WHERE thread.THREAD_ID = ?", + ThreadRowMapper(), + threadId + ) + return if (threadList.isEmpty()) { + Optional.empty() + } else { + Optional.of(threadList[0]) + } + } + + override fun findAll(): List { + return jdbcTemplate.query("SELECT * FROM THREAD", ThreadRowMapper()) + } + + override fun findById(id: Long): Optional { + val threadList = jdbcTemplate.query( + "SELECT * FROM THREAD AS thread WHERE thread.ID = ?", ThreadRowMapper(), id + ) + return if (threadList.isEmpty()) { + Optional.empty() + } else { + Optional.of(threadList[0]) + } + } + + override fun deleteByThreadId(threadId: String): Int { + val mutationCount = + jdbcTemplate.update("DELETE FROM THREAD AS thread WHERE THREAD_ID = ?", threadId) + eventBus.publishEvent(Event(Event.Kind.THREAD_REMOVE, threadId)) + return mutationCount + } + + override fun deleteAll() { + jdbcTemplate.update("DELETE FROM THREAD") + } +} + +internal class ThreadRowMapper : RowMapper { + @Throws(SQLException::class) + override fun mapRow(rs: ResultSet, rowNum: Int): Thread { + val id = rs.getLong("ID") + val link = rs.getString("LINK") + val threadId = rs.getString("THREAD_ID") + val total = rs.getInt("TOTAL") + return Thread(id, link, threadId, total) + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/DataTransaction.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/DataTransaction.kt new file mode 100644 index 00000000..bbe908b3 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/DataTransaction.kt @@ -0,0 +1,149 @@ +package me.mnlr.vripper.services + +import org.springframework.stereotype.Service +import org.springframework.transaction.annotation.Transactional +import me.mnlr.vripper.entities.ImageDownloadState +import me.mnlr.vripper.entities.Metadata +import me.mnlr.vripper.entities.PostDownloadState +import me.mnlr.vripper.entities.Thread +import me.mnlr.vripper.entities.domain.Status +import me.mnlr.vripper.model.PostItem +import me.mnlr.vripper.repositories.ImageRepository +import me.mnlr.vripper.repositories.MetadataRepository +import me.mnlr.vripper.repositories.PostDownloadStateRepository +import me.mnlr.vripper.repositories.ThreadRepository +import kotlin.io.path.pathString + +@Service +@Transactional +class DataTransaction( + private val postDownloadStateRepository: PostDownloadStateRepository, + private val imageRepository: ImageRepository, + private val threadRepository: ThreadRepository, + private val metadataRepository: MetadataRepository, + private val settingsService: SettingsService, + private val pathService: PathService +) { + private fun save(postDownloadState: PostDownloadState): PostDownloadState { + return postDownloadStateRepository.save(postDownloadState) + } + + fun update(postDownloadState: PostDownloadState) { + postDownloadStateRepository.update(postDownloadState) + } + + fun save(thread: Thread): Thread { + return threadRepository.save(thread) + } + + private fun save(imageDownloadState: ImageDownloadState): ImageDownloadState { + return imageRepository.save(imageDownloadState) + } + + fun update(imageDownloadState: ImageDownloadState) { + imageRepository.update(imageDownloadState) + } + + fun exists(postId: String): Boolean { + return postDownloadStateRepository.existByPostId(postId) + } + + fun newPost(postItem: PostItem): PostDownloadState { + val postDownloadState = save(PostDownloadState( + postTitle = postItem.title, + url = postItem.url, + token = postItem.securityToken, + postId = postItem.postId, + threadId = postItem.threadId, + total = postItem.imageCount, + hosts = postItem.hosts.keys.map { it.host }.toSet(), + threadTitle = postItem.threadTitle, + forum = postItem.forum, + downloadDirectory = pathService.calculateDownloadPath(postItem.forum, postItem.threadTitle, postItem.title, postItem.postId, settingsService.settings).pathString + )) + val images: MutableList = mutableListOf() + postItem.imageItemList.forEachIndexed { index, imageItem -> + val imageDownloadState = ImageDownloadState( + postId = postItem.postId, + url = imageItem.mainLink, + host = imageItem.host, + index = index, + postIdRef = postDownloadState.id!! + ) + images.add(save(imageDownloadState)) + } + sortPostsByRank() + return postDownloadState + } + + fun finishPost(postDownloadState: PostDownloadState) { + if (imageRepository.findByPostIdAndIsError(postDownloadState.postId).isNotEmpty()) { + postDownloadState.status = Status.ERROR + update(postDownloadState) + } else { + if (postDownloadState.done < postDownloadState.total) { + postDownloadState.status = Status.STOPPED + update(postDownloadState) + } else { + postDownloadState.status = Status.FINISHED + update(postDownloadState) + if (settingsService.settings.downloadSettings.clearCompleted) { + remove(listOf(postDownloadState.postId)) + } + } + } + } + + private fun remove(postIds: List) { + for (postId in postIds) { + imageRepository.deleteAllByPostId(postId) + metadataRepository.deleteByPostId(postId) + postDownloadStateRepository.deleteByPostId(postId) + } + sortPostsByRank() + } + + fun removeThread(threadId: String) { + threadRepository.deleteByThreadId(threadId) + } + + fun clearCompleted(): List { + val completed = postDownloadStateRepository.findCompleted() + remove(completed) + return completed + } + + fun removeAll(postIds: List?) { + if (postIds != null) { + remove(postIds) + } else { + remove(postDownloadStateRepository.findAll().map(PostDownloadState::postId)) + } + } + + fun stopImagesByPostIdAndIsNotCompleted(postId: String) { + imageRepository.stopByPostIdAndIsNotCompleted(postId) + } + + @Synchronized + fun setMetadata(postDownloadState: PostDownloadState, metadata: Metadata) { + if (metadataRepository.findByPostId(postDownloadState.postId).isEmpty) { + metadata.postIdRef = postDownloadState.id + metadataRepository.save(metadata) + } + } + + fun clearQueueLinks() { + threadRepository.deleteAll() + } + + @Synchronized + fun sortPostsByRank() { + val postDownloadState = + postDownloadStateRepository.findAll().sortedWith(Comparator.comparing(PostDownloadState::addedOn)) + for (i in postDownloadState.indices) { + postDownloadState[i].rank = i + update(postDownloadState[i]) + } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/DownloadSpeedService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/DownloadSpeedService.kt new file mode 100644 index 00000000..f1fe9edb --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/DownloadSpeedService.kt @@ -0,0 +1,30 @@ +package me.mnlr.vripper.services + +import org.springframework.scheduling.annotation.Scheduled +import org.springframework.stereotype.Service +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import java.util.concurrent.atomic.AtomicLong + +@Service +class DownloadSpeedService(private val eventBus: EventBus) { + private val read = AtomicLong(0) + var currentValue = 0L + private var allowWrite = false + fun increase(read: Long) { + if (allowWrite) { + this.read.addAndGet(read) + } + } + + @Scheduled(fixedDelay = 1000) + private fun calc() { + allowWrite = false + val newValue = read.getAndSet(0) + if (newValue != currentValue) { + currentValue = newValue + eventBus.publishEvent(Event(Event.Kind.BYTES_PER_SECOND, currentValue)) + } + allowWrite = true + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/GlobalStateService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/GlobalStateService.kt new file mode 100644 index 00000000..8419508b --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/GlobalStateService.kt @@ -0,0 +1,44 @@ +package me.mnlr.vripper.services + +import org.springframework.scheduling.annotation.Scheduled +import org.springframework.stereotype.Service +import me.mnlr.vripper.download.DownloadService +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.formatSI +import me.mnlr.vripper.model.GlobalState +import me.mnlr.vripper.repositories.ImageRepository + +@Service +class GlobalStateService( + private val downloadService: DownloadService, + private val imageRepository: ImageRepository, + private val eventBus: EventBus, + private val vgAuthService: VGAuthService, + private val downloadSpeedService: DownloadSpeedService +) { + private var currentState: GlobalState = newValue() + + @Scheduled(fixedDelay = 200) + private fun interval() { + val newGlobalState = newValue() + if (newGlobalState != currentState) { + currentState = newGlobalState + eventBus.publishEvent(Event(Event.Kind.DOWNLOAD_STATUS, currentState)) + } + } + + private fun newValue(): GlobalState { + return GlobalState( + downloadService.runningCount(), + downloadService.pendingCount(), + imageRepository.countError(), + vgAuthService.loggedUser, + downloadSpeedService.currentValue.formatSI() + ) + } + + fun get(): GlobalState { + return currentState.copy() + } +} diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/HTTPService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/HTTPService.kt new file mode 100644 index 00000000..19ec75ab --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/HTTPService.kt @@ -0,0 +1,131 @@ +package me.mnlr.vripper.services + +import jakarta.annotation.PreDestroy +import org.apache.http.client.config.CookieSpecs +import org.apache.http.client.config.RequestConfig +import org.apache.http.client.methods.AbstractExecutionAwareRequest +import org.apache.http.client.methods.HttpGet +import org.apache.http.client.methods.HttpHead +import org.apache.http.client.methods.HttpPost +import org.apache.http.client.protocol.HttpClientContext +import org.apache.http.impl.client.HttpClientBuilder +import org.apache.http.impl.client.HttpClients +import org.apache.http.impl.client.LaxRedirectStrategy +import org.apache.http.impl.conn.PoolingHttpClientConnectionManager +import org.springframework.scheduling.annotation.Scheduled +import org.springframework.stereotype.Service +import reactor.core.Disposable +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.model.Settings +import java.net.URI +import java.util.concurrent.TimeUnit + +@Service +class HTTPService(eventBus: EventBus, settingsService: SettingsService) { + private val disposable: Disposable + private lateinit var pcm: PoolingHttpClientConnectionManager + private lateinit var rc: RequestConfig + private var connectionTimeout: Int = settingsService.settings.connectionSettings.timeout + + init { + disposable = eventBus + .flux() + .filter { it.kind == Event.Kind.SETTINGS_UPDATE } + .map { it.data as Settings } + .subscribe { + if (connectionTimeout != it.connectionSettings.timeout) { + connectionTimeout = it.connectionSettings.timeout + buildRequestConfig() + } + } + buildRequestConfig() + buildConnectionPool() + } + + @PreDestroy + private fun destroy() { + disposable.dispose() + } + + private fun buildConnectionPool() { + pcm = PoolingHttpClientConnectionManager() + pcm.maxTotal = Int.MAX_VALUE + pcm.defaultMaxPerRoute = Int.MAX_VALUE + } + + private fun buildRequestConfig() { + rc = RequestConfig.custom() + .setConnectionRequestTimeout(connectionTimeout * 1000) + .setConnectTimeout(connectionTimeout * 1000) + .setSocketTimeout(connectionTimeout * 1000) + .setCookieSpec(CookieSpecs.STANDARD) + .build() + } + + @Scheduled(fixedDelay = 15000) + private fun idleConnectionMonitoring() { + pcm.closeIdleConnections(60, TimeUnit.SECONDS) + } + + val client: HttpClientBuilder + get() = HttpClients.custom() + .setConnectionManager(pcm) + .setRedirectStrategy(LaxRedirectStrategy()) + .disableAutomaticRetries() + .setDefaultRequestConfig(rc) + + fun buildHttpGet(url: String, context: HttpClientContext): HttpGet { + val httpGet = HttpGet(url.replace(" ", "+")) + httpGet.addHeader("User-Agent", USER_AGENT) + addToContext(context, httpGet) + return httpGet + } + + fun buildHttpHead(url: String, context: HttpClientContext): HttpHead { + val httpHead = HttpHead(url.replace(" ", "+")) + httpHead.addHeader("User-Agent", USER_AGENT) + addToContext(context, httpHead) + return httpHead + } + + fun buildHttpPost(url: String, context: HttpClientContext): HttpPost { + val httpPost = HttpPost(url.replace(" ", "+")) + httpPost.addHeader("User-Agent", USER_AGENT) + addToContext(context, httpPost) + return httpPost + } + + fun buildHttpGet(uri: URI, context: HttpClientContext): HttpGet { + val httpGet = HttpGet(uri) + httpGet.addHeader("User-Agent", USER_AGENT) + addToContext(context, httpGet) + return httpGet + } + + fun addToContext(context: HttpClientContext, request: AbstractExecutionAwareRequest) { + val contextAttributes = + context.getAttribute( + ContextAttributes.CONTEXT_ATTRIBUTES, + ContextAttributes::class.java + ) + if (contextAttributes != null) { + synchronized(contextAttributes.requests) { + contextAttributes.requests.add(request) + } + } + } + + class ContextAttributes { + val requests: MutableList = mutableListOf() + + companion object { + const val CONTEXT_ATTRIBUTES = "CONTEXT_ATTRIBUTES" + } + } + + companion object { + const val USER_AGENT = + "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36" + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/HtmlProcessorService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/HtmlProcessorService.kt new file mode 100644 index 00000000..72e066b4 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/HtmlProcessorService.kt @@ -0,0 +1,24 @@ +package me.mnlr.vripper.services + +import org.htmlcleaner.CleanerProperties +import org.htmlcleaner.DomSerializer +import org.htmlcleaner.HtmlCleaner +import org.springframework.stereotype.Service +import org.w3c.dom.Document +import me.mnlr.vripper.exception.HtmlProcessorException +import java.io.InputStream + +@Service +class HtmlProcessorService { + @Throws(HtmlProcessorException::class) + fun clean(htmlContent: InputStream): Document { + return try { + htmlContent.use { + val clean = HtmlCleaner().clean(htmlContent) + DomSerializer(CleanerProperties()).createDOM(clean) + } + } catch (e: Exception) { + throw HtmlProcessorException(e) + } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/MetadataService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/MetadataService.kt new file mode 100644 index 00000000..ac7252a4 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/MetadataService.kt @@ -0,0 +1,38 @@ +package me.mnlr.vripper.services + +import org.springframework.stereotype.Service +import me.mnlr.vripper.entities.PostDownloadState +import me.mnlr.vripper.tasks.MetadataRunnable +import java.util.function.Consumer + +@Service +class MetadataService(private val threadPoolService: ThreadPoolService) { + private val fetchingMetadata: MutableMap = mutableMapOf() + + @Synchronized + fun startFetchingMetadata(postDownloadState: PostDownloadState) { + val runnable = MetadataRunnable(postDownloadState) + threadPoolService.generalExecutor.submit(runnable) + fetchingMetadata[postDownloadState.postId] = runnable + } + + @Synchronized + fun stopFetchingMetadata(postIds: List) { + val stopping: MutableList = ArrayList() + for ((key, value) in fetchingMetadata) { + if (postIds.contains(key)) { + stopping.add(value) + value.stop() + } + } + while (stopping.isNotEmpty()) { + stopping.removeIf { it.finished } + try { + Thread.sleep(100) + } catch (e: InterruptedException) { + Thread.currentThread().interrupt() + } + } + postIds.forEach(Consumer { key: String -> fetchingMetadata.remove(key) }) + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/PathService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/PathService.kt new file mode 100644 index 00000000..345e7bc8 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/PathService.kt @@ -0,0 +1,49 @@ +package me.mnlr.vripper.services + +import org.springframework.stereotype.Service +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.model.Settings +import java.io.File +import java.nio.file.Path + +@Service +class PathService { + private val log by LoggerDelegate() + + fun calculateDownloadPath(forum: String, threadTitle: String, postTitle: String, postId: String, settings: Settings): Path { + var downloadDirectory = + if (settings.downloadSettings.forumSubfolder) Path.of(settings.downloadSettings.downloadPath, sanitize(forum)) else Path.of( + settings.downloadSettings.downloadPath + ) + downloadDirectory = if (settings.downloadSettings.threadSubLocation) downloadDirectory.resolve(threadTitle) else downloadDirectory + downloadDirectory = downloadDirectory.resolve(if (settings.downloadSettings.appendPostId) "${sanitize(postTitle)}_${postId}" else sanitize( + postTitle + )) + return downloadDirectory + } + + private fun makeDir(sourceFolder: File): File { + var counter = 1 + var folder = sourceFolder + while (folder.exists()) { + folder = File(sourceFolder.toString() + '.' + counter++) + } + if (!folder.mkdirs()) { + throw RuntimeException(String.format("Failed to create the folder %s", sourceFolder)) + } + return folder + } + + /** + * Will sanitize the image name and remove extension + * + * @param path + * @return Sanitized local path string + */ + fun sanitize(path: String): String { + val sanitizedPath = + path.replace("\\.|\\\\|/|\\||:|\\?|\\*|\"|<|>|\\p{Cntrl}".toRegex(), "_") + log.debug(String.format("%s sanitized to %s", path, sanitizedPath)) + return sanitizedPath + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/RetryPolicyService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/RetryPolicyService.kt new file mode 100644 index 00000000..70489f17 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/RetryPolicyService.kt @@ -0,0 +1,54 @@ +package me.mnlr.vripper.services + +import jakarta.annotation.PreDestroy +import net.jodah.failsafe.RetryPolicy +import net.jodah.failsafe.event.ExecutionAttemptedEvent +import org.springframework.stereotype.Service +import reactor.core.Disposable +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.model.Settings +import java.time.temporal.ChronoUnit + +@Service +class RetryPolicyService(eventBus: EventBus, settingsService: SettingsService) { + private val log by LoggerDelegate() + private val disposable: Disposable + private var maxAttempts: Int = settingsService.settings.connectionSettings.maxAttempts + + init { + disposable = eventBus + .flux() + .filter { it.kind == Event.Kind.SETTINGS_UPDATE } + .map { it.data as Settings } + .subscribe { + if (maxAttempts != it.connectionSettings.maxAttempts) { + maxAttempts = it.connectionSettings.maxAttempts + } + } + } + + fun buildRetryPolicyForDownload(): RetryPolicy { + return RetryPolicy() + .withDelay(2, 5, ChronoUnit.SECONDS) + .withMaxAttempts(maxAttempts) + .onFailedAttempt { + log.warn("#${it.attemptCount} tries failed", it.lastFailure) + } + } + + fun buildGenericRetryPolicy(): RetryPolicy { + return RetryPolicy() + .withDelay(2, 5, ChronoUnit.SECONDS) + .withMaxAttempts(maxAttempts) + .onFailedAttempt { e: ExecutionAttemptedEvent -> + log.warn("#${e.attemptCount} tries failed", e.lastFailure) + } + } + + @PreDestroy + private fun destroy() { + disposable.dispose() + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/SettingsService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/SettingsService.kt new file mode 100644 index 00000000..3eda73de --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/SettingsService.kt @@ -0,0 +1,211 @@ +package me.mnlr.vripper.services + +import com.fasterxml.jackson.databind.DeserializationFeature +import com.fasterxml.jackson.databind.ObjectMapper +import com.fasterxml.jackson.dataformat.yaml.YAMLFactory +import com.fasterxml.jackson.module.kotlin.readValue +import com.fasterxml.jackson.module.kotlin.registerKotlinModule +import jakarta.annotation.PostConstruct +import jakarta.annotation.PreDestroy +import org.apache.commons.codec.digest.DigestUtils +import org.springframework.beans.factory.annotation.Value +import org.springframework.core.io.Resource +import org.springframework.stereotype.Service +import me.mnlr.vripper.SpringContext +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.exception.ValidationException +import me.mnlr.vripper.model.Settings +import java.io.FileWriter +import java.io.IOException +import java.nio.file.* + +@Service +class SettingsService( + @Value("\${base.dir}") baseDir: String, + @Value("\${base.dir.name}") baseDirName: String, + private val eventBus: EventBus, + @Value("classpath:proxies.json") private val defaultProxies: Resource +) { + private val log by LoggerDelegate() + private val configPath: Path + private val customProxiesPath: Path + private val om = ObjectMapper(YAMLFactory()) + private val proxies: MutableSet = HashSet() + + + var settings = Settings() + + init { + configPath = Paths.get(baseDir, baseDirName, "config.yml") + customProxiesPath = Paths.get(baseDir, baseDirName, "proxies.json") + om.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false) + .findAndRegisterModules().registerKotlinModule() + } + + @PostConstruct + private fun init() { + loadViperProxies() + restore() + eventBus.publishEvent(Event(Event.Kind.SETTINGS_UPDATE, settings)) + } + + private fun loadViperProxies() { + try { + defaultProxies.inputStream.use { defaultProxiesIs -> + val defaultProxies: List = om.readValue(defaultProxiesIs) + val customProxies: List = if (customProxiesPath.toFile() + .exists() && Files.isRegularFile(customProxiesPath) + ) { + om.readValue( + customProxiesPath.toFile() + ) + } else { + emptyList() + } + + try { + FileWriter(customProxiesPath.toFile()).use { fw -> + fw.append("[]").append(System.lineSeparator()) + fw.flush() + } + } catch (e: IOException) { + log.warn( + "Unable to create " + customProxiesPath.toFile().absolutePath, e + ) + } + + proxies.addAll(customProxies) + proxies.addAll(defaultProxies) + } + } catch (e: IOException) { + log.error("Failed to load vipergirls proxies list", e) + } + proxies.add("https://vipergirls.to") + } + + fun getProxies(): List { + return ArrayList(proxies) + } + + fun newSettings(settings: Settings) { + if (settings.viperSettings.login) { + if (this.settings.viperSettings.password != settings.viperSettings.password) { + settings.viperSettings.password = DigestUtils.md5Hex(settings.viperSettings.password) + } + } else { + settings.viperSettings.username = "" + settings.viperSettings.password = "" + settings.viperSettings.thanks = false + settings.viperSettings.login = false + } + check(settings) + this.settings = settings + save() + eventBus.publishEvent(Event(Event.Kind.SETTINGS_UPDATE, settings)) + } + + fun restore() { + try { + if (configPath.toFile().exists()) { + settings = om.readValue(configPath.toFile()) + } + } catch (e: IOException) { + log.error("Failed restore user settings", e) + settings = Settings() + } + if (!proxies.contains(settings.viperSettings.host)) { + settings.viperSettings.host = "https://vipergirls.to" + } + try { + check(settings) + } catch (e: ValidationException) { + log.error( + String.format( + "Your settings are invalid, either remove %s, or fix it", configPath + ), e + ) + SpringContext.close() + } + save() + } + + fun save() { + try { + Files.write( + configPath, + om.writeValueAsBytes(settings), + StandardOpenOption.CREATE, + StandardOpenOption.WRITE, + StandardOpenOption.TRUNCATE_EXISTING, + StandardOpenOption.SYNC + ) + } catch (e: IOException) { + log.error("Failed to store user settings", e) + } + } + + @PreDestroy + private fun destroy() { + save() + } + + @Throws(ValidationException::class) + fun check(settings: Settings) { + val path: Path = try { + Paths.get(settings.downloadSettings.downloadPath) + } catch (e: InvalidPathException) { + throw ValidationException(String.format("%s is invalid", settings.downloadSettings.downloadPath)) + } + if (!Files.exists(path)) { + throw ValidationException(String.format("%s does not exist", settings.downloadSettings.downloadPath)) + } else if (!Files.isDirectory(path)) { + throw ValidationException(String.format("%s is not a directory", settings.downloadSettings.downloadPath)) + } + + if(settings.downloadSettings.autoQueueThreshold < 0) { + throw ValidationException("Invalid auto queue settings, value must be a positive integer") + } + + if (settings.connectionSettings.maxTotalThreads < 0 || settings.connectionSettings.maxTotalThreads > 12) { + throw ValidationException( + String.format( + "Invalid max global concurrent download settings, values must be in [%d,%d]", + 0, + 12 + ) + ) + } + if (settings.connectionSettings.maxThreads < 1 || settings.connectionSettings.maxThreads > 4) { + throw ValidationException( + String.format( + "Invalid max concurrent download settings, values must be in [%d,%d]", 1, 4 + ) + ) + } + if (settings.connectionSettings.timeout < 1 || settings.connectionSettings.timeout > 300) { + throw ValidationException( + String.format( + "Invalid connection timeout settings, values must be in [%d,%d]", 1, 300 + ) + ) + } + if (settings.connectionSettings.maxAttempts < 1 || settings.connectionSettings.maxAttempts > 10) { + throw ValidationException( + String.format( + "Invalid maximum attempts settings, values must be in [%d,%d]", 1, 10 + ) + ) + } + if (settings.maxEventLog < 100 || settings.maxEventLog > 10000) { + throw ValidationException( + String.format( + "Invalid maximum event log record settings, values must be in [%d,%d]", + 100, + 10000 + ) + ) + } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/ThreadCacheService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/ThreadCacheService.kt new file mode 100644 index 00000000..c44e17a5 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/ThreadCacheService.kt @@ -0,0 +1,39 @@ +package me.mnlr.vripper.services + +import com.github.benmanes.caffeine.cache.Caffeine +import com.github.benmanes.caffeine.cache.LoadingCache +import jakarta.annotation.PreDestroy +import org.springframework.stereotype.Service +import reactor.core.Disposable +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.model.ThreadItem +import me.mnlr.vripper.parser.ThreadLookupAPIParser +import java.util.* +import java.util.concurrent.ExecutionException +import java.util.concurrent.TimeUnit + +@Service +class ThreadCacheService(eventBus: EventBus) { + private val cache: LoadingCache = + Caffeine.newBuilder().expireAfterWrite(5, TimeUnit.MINUTES) + .build { threadId: String -> + ThreadLookupAPIParser(threadId).parse() + } + private val disposable: Disposable + + init { + disposable = eventBus.flux().filter { e: Event<*> -> e.kind == Event.Kind.SETTINGS_UPDATE } + .subscribe { cache.invalidateAll() } + } + + @PreDestroy + private fun destroy() { + disposable.dispose() + } + + @Throws(ExecutionException::class) + operator fun get(threadId: String): ThreadItem { + return cache[threadId] ?: throw NoSuchElementException("$threadId does not exist") + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/ThreadPoolService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/ThreadPoolService.kt new file mode 100644 index 00000000..8be9b8e1 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/ThreadPoolService.kt @@ -0,0 +1,17 @@ +package me.mnlr.vripper.services + +import org.springframework.stereotype.Service +import java.util.concurrent.ExecutorService +import java.util.concurrent.Executors +import java.util.concurrent.TimeUnit + +@Service +class ThreadPoolService { + val generalExecutor: ExecutorService = Executors.newFixedThreadPool(4) + fun destroy() { + generalExecutor.shutdown() + if (!generalExecutor.awaitTermination(5, TimeUnit.SECONDS)) { + generalExecutor.shutdownNow() + } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/VGAuthService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/VGAuthService.kt new file mode 100644 index 00000000..3e5c2cc3 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/VGAuthService.kt @@ -0,0 +1,142 @@ +package me.mnlr.vripper.services + +import jakarta.annotation.PostConstruct +import jakarta.annotation.PreDestroy +import org.apache.http.NameValuePair +import org.apache.http.client.entity.UrlEncodedFormEntity +import org.apache.http.client.protocol.HttpClientContext +import org.apache.http.cookie.Cookie +import org.apache.http.impl.client.BasicCookieStore +import org.apache.http.message.BasicNameValuePair +import org.apache.http.util.EntityUtils +import org.springframework.stereotype.Service +import reactor.core.Disposable +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.entities.PostDownloadState +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.exception.VripperException +import me.mnlr.vripper.model.PostItem +import me.mnlr.vripper.tasks.LeaveThanksRunnable + +@Service +class VGAuthService( + private val cm: HTTPService, + private val settingsService: SettingsService, + private val threadPoolService: ThreadPoolService, + private val eventBus: EventBus +) { + private val log by LoggerDelegate() + private val disposable: Disposable + + final val context: HttpClientContext = HttpClientContext.create() + var loggedUser = "" + + var authenticated = false + + init { + context.cookieStore = BasicCookieStore() + disposable = eventBus + .flux() + .filter { it.kind == Event.Kind.SETTINGS_UPDATE } + .subscribe { authenticate() } + } + + @PostConstruct + private fun init() { + authenticate() + } + + @PreDestroy + private fun destroy() { + disposable.dispose() + } + + fun authenticate() { + authenticated = false + if (!settingsService.settings.viperSettings.login) { + log.debug("Authentication option is disabled") + context.cookieStore.clear() + loggedUser = "" + eventBus.publishEvent(Event(Event.Kind.VG_USER, loggedUser)) + return + } + val username = settingsService.settings.viperSettings.username + val password = settingsService.settings.viperSettings.password + if (username.isEmpty() || password.isEmpty()) { + log.error("Cannot authenticate with ViperGirls credentials, username or password is empty") + context.cookieStore.clear() + loggedUser = "" + eventBus.publishEvent(Event(Event.Kind.VG_USER, loggedUser)) + return + } + val postAuth = + cm.buildHttpPost(settingsService.settings.viperSettings.host + "/login.php?do=login", context) + val params: MutableList = ArrayList() + params.add(BasicNameValuePair("vb_login_username", username)) + params.add(BasicNameValuePair("cookieuser", "1")) + params.add(BasicNameValuePair("do", "login")) + params.add(BasicNameValuePair("vb_login_md5password", password)) + try { + postAuth.entity = UrlEncodedFormEntity(params) + } catch (e: Exception) { + context.cookieStore.clear() + loggedUser = "" + eventBus.publishEvent(Event(Event.Kind.VG_USER, loggedUser)) + log.error("Failed to authenticate user with " + settingsService.settings.viperSettings.host, e) + return + } + postAuth.addHeader("Referer", settingsService.settings.viperSettings.host) + postAuth.addHeader( + "Host", + settingsService.settings.viperSettings.host.replace("https://", "").replace("http://", "") + ) + val client = cm.client.build() + try { + client.execute(postAuth, context).use { response -> + if (response.statusLine.statusCode / 100 != 2) { + throw VripperException( + String.format( + "Unexpected response code returned %s", response.statusLine.statusCode + ) + ) + } + val responseBody = EntityUtils.toString(response.entity) + log.debug( + String.format( + "Authentication with ViperGirls response body:%n%s", + responseBody + ) + ) + EntityUtils.consumeQuietly(response.entity) + if (context.cookieStore.cookies.stream() + .map { obj: Cookie -> obj.name } + .noneMatch { e: String -> e == "vg_userid" } + ) { + log.error( + String.format( + "Failed to authenticate user with %s, missing vg_userid cookie", + settingsService.settings.viperSettings.host + ) + ) + return + } + } + } catch (e: Exception) { + context.cookieStore.clear() + loggedUser = "" + eventBus.publishEvent(Event(Event.Kind.VG_USER, loggedUser)) + log.error("Failed to authenticate user with " + settingsService.settings.viperSettings.host, e) + return + } + authenticated = true + loggedUser = username + log.info(String.format("Authenticated: %s", username)) + eventBus.publishEvent(Event(Event.Kind.VG_USER, loggedUser)) + } + + fun leaveThanks(post: PostDownloadState) { + threadPoolService.generalExecutor + .submit(LeaveThanksRunnable(post, authenticated, context)) + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/services/XpathService.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/XpathService.kt new file mode 100644 index 00000000..d2f52a57 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/services/XpathService.kt @@ -0,0 +1,31 @@ +package me.mnlr.vripper.services + +import org.springframework.stereotype.Service +import org.w3c.dom.Node +import org.w3c.dom.NodeList +import me.mnlr.vripper.exception.XpathException +import javax.xml.xpath.XPathConstants +import javax.xml.xpath.XPathFactory + +@Service +class XpathService { + private val xpath = XPathFactory.newInstance().newXPath() + + @Throws(XpathException::class) + fun getAsNode(source: Node?, xpathExpression: String?): Node? { + return try { + xpath.compile(xpathExpression).evaluate(source, XPathConstants.NODE) as Node + } catch (e: Exception) { + throw XpathException(e) + } + } + + @Throws(XpathException::class) + fun getAsNodeList(source: Node?, xpathExpression: String?): NodeList { + return try { + xpath.compile(xpathExpression).evaluate(source, XPathConstants.NODESET) as NodeList + } catch (e: Exception) { + throw XpathException(e) + } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/LeaveThanksRunnable.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/LeaveThanksRunnable.kt new file mode 100644 index 00000000..75824bb3 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/LeaveThanksRunnable.kt @@ -0,0 +1,118 @@ +package me.mnlr.vripper.tasks + +import org.apache.http.NameValuePair +import org.apache.http.client.entity.UrlEncodedFormEntity +import org.apache.http.client.methods.HttpPost +import org.apache.http.client.protocol.HttpClientContext +import org.apache.http.impl.client.CloseableHttpClient +import org.apache.http.message.BasicNameValuePair +import org.apache.http.util.EntityUtils +import me.mnlr.vripper.SpringContext +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.entities.LogEvent +import me.mnlr.vripper.entities.LogEvent.Status.* +import me.mnlr.vripper.entities.PostDownloadState +import me.mnlr.vripper.formatToString +import me.mnlr.vripper.repositories.LogEventRepository +import me.mnlr.vripper.services.HTTPService +import me.mnlr.vripper.services.SettingsService +import java.io.UnsupportedEncodingException + +class LeaveThanksRunnable( + private val postDownloadState: PostDownloadState, + private val authenticated: Boolean, + private val context: HttpClientContext +) : Runnable { + private val log by LoggerDelegate() + private val cm: HTTPService = SpringContext.getBean(HTTPService::class.java) + private val settingsService: SettingsService = SpringContext.getBean(SettingsService::class.java) + private val eventRepository: LogEventRepository = SpringContext.getBean(LogEventRepository::class.java) + private val logEvent: LogEvent + + init { + logEvent = eventRepository.save( + LogEvent( + type = LogEvent.Type.THANKS, status = PENDING, message = "Leaving thanks for $postDownloadState.url" + ) + ) + } + + override fun run() { + try { + eventRepository.update(logEvent.copy(status = PROCESSING)) + if (!settingsService.settings.viperSettings.login) { + eventRepository.update( + logEvent.copy( + status = DONE, + message = "Will not send a like for ${postDownloadState.url}\nAuthentication with ViperGirls option is disabled" + ) + ) + return + } + if (!settingsService.settings.viperSettings.thanks) { + eventRepository.update( + logEvent.copy( + status = DONE, + message = "Will not send a like for ${postDownloadState.url}\nLeave thanks option is disabled" + ) + ) + return + } + if (!authenticated) { + eventRepository.update( + logEvent.copy( + status = ERROR, + message = "Will not send a like for ${postDownloadState.url}\nYou are not authenticated" + ) + ) + return + } + val postThanks: HttpPost = cm.buildHttpPost( + "${settingsService.settings.viperSettings.host}/post_thanks.php", HttpClientContext.create() + ) + val params: MutableList = ArrayList() + params.add(BasicNameValuePair("do", "post_thanks_add")) + params.add(BasicNameValuePair("using_ajax", "1")) + params.add(BasicNameValuePair("p", postDownloadState.postId)) + params.add(BasicNameValuePair("securitytoken", postDownloadState.token)) + try { + postThanks.entity = UrlEncodedFormEntity(params) + } catch (e: UnsupportedEncodingException) { + val error = "Request error for ${postDownloadState.url}" + log.error(error, e) + eventRepository.update( + logEvent.copy( + status = ERROR, message = """ + $error + ${e.formatToString()} + """.trimIndent() + ) + ) + return + } + postThanks.addHeader("Referer", settingsService.settings.viperSettings.host) + postThanks.addHeader( + "Host", settingsService.settings.viperSettings.host.replace("https://", "").replace("http://", "") + ) + val client: CloseableHttpClient = cm.client.build() + client.execute(postThanks, context).use { response -> + try { + } finally { + EntityUtils.consumeQuietly(response.entity) + } + } + eventRepository.update(logEvent.copy(status = DONE)) + } catch (e: Exception) { + val error = "Failed to leave a thanks for $postDownloadState" + log.error(error, e) + eventRepository.update( + logEvent.copy( + status = ERROR, message = """ + $error + ${e.formatToString()} + """.trimIndent() + ) + ) + } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/LinkScanRunnable.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/LinkScanRunnable.kt new file mode 100644 index 00000000..33517fff --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/LinkScanRunnable.kt @@ -0,0 +1,110 @@ +package me.mnlr.vripper.tasks + +import me.mnlr.vripper.SpringContext +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.entities.LogEvent +import me.mnlr.vripper.entities.LogEvent.Status.* +import me.mnlr.vripper.formatToString +import me.mnlr.vripper.model.ThreadInput +import me.mnlr.vripper.repositories.impl.LogEventRepositoryImpl +import me.mnlr.vripper.services.SettingsService +import me.mnlr.vripper.services.ThreadPoolService +import java.time.LocalDateTime +import java.util.regex.Pattern + +class LinkScanRunnable(private val urlList: List) : Runnable { + private val log by LoggerDelegate() + private val settingsService: SettingsService = + SpringContext.getBean(SettingsService::class.java) + private val threadPoolService: ThreadPoolService = + SpringContext.getBean(ThreadPoolService::class.java) + private val eventRepository: LogEventRepositoryImpl = + SpringContext.getBean(LogEventRepositoryImpl::class.java) + private val logEvent: LogEvent + + init { + logEvent = eventRepository.save( + LogEvent( + type = LogEvent.Type.SCAN, + status = PENDING, + time = LocalDateTime.now(), + message = "Links to scan:\n\t${urlList.joinToString("\n\t")}" + ) + ) + } + + override fun run() { + try { + eventRepository.update(logEvent.copy(status = PROCESSING)) + val threadInputList = mutableListOf() + val unsupported = mutableListOf() + val unrecognized = mutableListOf() + for (link in urlList) { + log.debug("Starting to process thread: $link") + if (!link.startsWith(settingsService.settings.viperSettings.host)) { + log.error("Unsupported link $link") + unsupported.add(link) + continue + } + var threadId: String + val m = Pattern.compile( + Pattern.quote(settingsService.settings.viperSettings.host) + "/threads/(\\d+).*" + ).matcher(link) + if (m.find()) { + threadId = m.group(1) + } else { + log.error("Cannot retrieve thread id from URL $link") + unrecognized.add(link) + continue + } + threadInputList.add(ThreadInput(link = link, threadId = threadId)) + } + val errorMessage: String = if (unsupported.isNotEmpty()) { + """Unsupported links: + ${unsupported.joinToString("\n\t")} + """ + } else if (unrecognized.isNotEmpty()) { + """Unrecognized links: + ${unrecognized.joinToString("\n\t")} + """ + } else { + "" + } + if (errorMessage.isNotBlank()) { + eventRepository.update( + logEvent.copy( + status = ERROR, message = "Some links failed to be scanned: \n$errorMessage" + ) + ) + } + for (threadInput in threadInputList) { +// if (thread.postId.isNotBlank()) { +// threadPoolService.generalExecutor.submit( +// SinglePostRunnable( +// thread.threadId, thread.postId +// ) +// ) +// } else { + threadPoolService.generalExecutor.submit( + ThreadLookupRunnable( + threadInput, + settingsService.settings + ) + ) +// } + } + eventRepository.update(logEvent.copy(status = DONE)) + } catch (e: Exception) { + val error = "Error when scanning links" + log.error(error, e) + eventRepository.update( + logEvent.copy( + status = ERROR, message = """ + $error + ${e.formatToString()} + """.trimIndent() + ) + ) + } + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/MetadataRunnable.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/MetadataRunnable.kt new file mode 100644 index 00000000..d32a7d94 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/MetadataRunnable.kt @@ -0,0 +1,177 @@ +package me.mnlr.vripper.tasks + +import org.apache.http.client.HttpClient +import org.apache.http.client.methods.CloseableHttpResponse +import org.apache.http.client.methods.HttpGet +import org.apache.http.client.protocol.HttpClientContext +import org.apache.http.util.EntityUtils +import org.w3c.dom.Document +import org.w3c.dom.Node +import me.mnlr.vripper.SpringContext +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.entities.LogEvent +import me.mnlr.vripper.entities.LogEvent.Status.* +import me.mnlr.vripper.entities.Metadata +import me.mnlr.vripper.entities.PostDownloadState +import me.mnlr.vripper.exception.DownloadException +import me.mnlr.vripper.formatToString +import me.mnlr.vripper.repositories.LogEventRepository +import me.mnlr.vripper.services.* +import java.util.concurrent.atomic.AtomicBoolean +import java.util.stream.Collectors + +class MetadataRunnable(private val postDownloadState: PostDownloadState) : Runnable { + + private val log by LoggerDelegate() + private val dataTransaction: DataTransaction = + SpringContext.getBean(DataTransaction::class.java) + private val eventRepository: LogEventRepository = + SpringContext.getBean(LogEventRepository::class.java) + private val cm: HTTPService = SpringContext.getBean(HTTPService::class.java) + private val htmlProcessorService: HtmlProcessorService = + SpringContext.getBean(HtmlProcessorService::class.java) + private val xpathService: XpathService = SpringContext.getBean(XpathService::class.java) + private val metadataService: MetadataService = + SpringContext.getBean(MetadataService::class.java) + private val context: HttpClientContext = HttpClientContext.create() + private val logEvent: LogEvent + + @Volatile + private var stopped = false + + @Volatile + var finished = false + + init { + val vgAuthService: VGAuthService = SpringContext.getBean(VGAuthService::class.java) + context.cookieStore = vgAuthService.context.cookieStore + context.setAttribute( + HTTPService.ContextAttributes.CONTEXT_ATTRIBUTES, HTTPService.ContextAttributes() + ) + logEvent = eventRepository.save( + LogEvent( + type = LogEvent.Type.METADATA, + status = PENDING, + message = "Fetching metadata for " + postDownloadState.url + ) + ) + } + + override fun run() { + try { + eventRepository.update(logEvent.copy(status = PROCESSING)) + val metadata = fetchMetadata(postDownloadState.postId, postDownloadState.url) + dataTransaction.setMetadata(postDownloadState, metadata) + eventRepository.update(logEvent.copy(status = DONE)) + } catch (e: Exception) { + val message = "Failed to fetch metadata for ${postDownloadState.url}" + log.error(message, e) + eventRepository.update( + logEvent.copy( + status = ERROR, message = """ + $message + ${e.formatToString()} + """.trimIndent() + ) + ) + } finally { + if (stopped) { + val message = "Fetching metadata for ${postDownloadState.url} interrupted" + eventRepository.update(logEvent.copy(status = DONE, message = message)) + } + finished = true + metadataService.stopFetchingMetadata(listOf(postDownloadState.postId)) + } + } + + private fun fetchMetadata(postId: String, url: String): Metadata { + val httpGet: HttpGet = cm.buildHttpGet(url, context) + val connection: HttpClient = cm.client.build() + (connection.execute(httpGet, context) as CloseableHttpResponse).use { response -> + try { + if (response.statusLine.statusCode / 100 != 2) { + throw DownloadException("Unexpected response code '${response.statusLine.statusCode}' for $httpGet") + } + val document: Document = + htmlProcessorService.clean(response.entity.content) + val postNode: Node? = + xpathService.getAsNode( + document, + "//li[@id='post_$postId']/div[contains(@class, 'postdetails')]" + ) + val postedBy: String = xpathService.getAsNode( + postNode, + "./div[contains(@class, 'userinfo')]//a[contains(@class, 'username')]//font" + )?.textContent?.trim() ?: "" + + val metadata = Metadata() + metadata.postedBy = postedBy + + val node: Node? = xpathService.getAsNode( + document, "//div[@id='post_message_$postId']" + ) + if (node != null) { + metadata.resolvedNames = findTitleInContent(node) + } else { + metadata.resolvedNames = emptyList() + } + metadata.postId = postId + return metadata + } finally { + EntityUtils.consumeQuietly(response.entity) + } + } + } + + private fun findTitleInContent(node: Node): List { + val altTitle: MutableList = ArrayList() + findTitle(node, altTitle, AtomicBoolean(true)) + return altTitle.stream().distinct().collect(Collectors.toList()) + } + + private fun findTitle(node: Node, altTitle: MutableList, keepGoing: AtomicBoolean) { + if (!keepGoing.get()) { + return + } + if (node.nodeName == "a" || node.nodeName == "img") { + keepGoing.set(false) + return + } + if (node.nodeType == Node.ELEMENT_NODE) { + for (i in 0 until node.childNodes.length) { + val item = node.childNodes.item(i) + findTitle(item, altTitle, keepGoing) + if (!keepGoing.get()) { + return + } + } + } else if (node.nodeType == Node.TEXT_NODE) { + val text = node.textContent.trim() + if (text.isNotBlank() && dictionary.stream().noneMatch { e: String -> + text.lowercase().contains(e.lowercase()) + }) { + altTitle.add(text) + } + } + } + + fun stop() { + stopped = true + val contextAttributes: HTTPService.ContextAttributes? = context.getAttribute( + HTTPService.ContextAttributes.CONTEXT_ATTRIBUTES, + HTTPService.ContextAttributes::class.java + ) + if (contextAttributes != null) { + synchronized(contextAttributes.requests) { + for (request in contextAttributes.requests) { + request.abort() + } + } + } + } + + companion object { + private val dictionary: List = + mutableListOf("download", "link", "rapidgator", "filefactory", "filefox") + } +} \ No newline at end of file diff --git a/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/ThreadLookupRunnable.kt b/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/ThreadLookupRunnable.kt new file mode 100644 index 00000000..880d63c5 --- /dev/null +++ b/vripper-core/src/main/kotlin/me/mnlr/vripper/tasks/ThreadLookupRunnable.kt @@ -0,0 +1,106 @@ +package me.mnlr.vripper.tasks + +import me.mnlr.vripper.SpringContext +import me.mnlr.vripper.delegate.LoggerDelegate +import me.mnlr.vripper.download.DownloadService +import me.mnlr.vripper.entities.LogEvent +import me.mnlr.vripper.entities.LogEvent.Status.* +import me.mnlr.vripper.entities.Thread +import me.mnlr.vripper.formatToString +import me.mnlr.vripper.model.Settings +import me.mnlr.vripper.model.ThreadInput +import me.mnlr.vripper.model.ThreadItem +import me.mnlr.vripper.repositories.LogEventRepository +import me.mnlr.vripper.repositories.ThreadRepository +import me.mnlr.vripper.services.DataTransaction +import me.mnlr.vripper.services.SettingsService +import me.mnlr.vripper.services.ThreadCacheService + +class ThreadLookupRunnable(private val threadInput: ThreadInput, private val settings: Settings) : Runnable { + private val log by LoggerDelegate() + private val dataTransaction = SpringContext.getBean(DataTransaction::class.java) + private val eventRepository = SpringContext.getBean(LogEventRepository::class.java) + private val threadCacheService = SpringContext.getBean( + ThreadCacheService::class.java + ) + private val threadRepository = SpringContext.getBean( + ThreadRepository::class.java + ) + private val downloadService = SpringContext.getBean(DownloadService::class.java) + private val logEvent: LogEvent + + init { + logEvent = eventRepository.save( + LogEvent( + type = LogEvent.Type.THREAD, + status = LogEvent.Status.PENDING, + message = "Processing multi-post link ${threadInput.link}" + ) + ) + } + + override fun run() { + try { + eventRepository.update(logEvent.copy(status = PROCESSING)) + val threadLookupResult = threadCacheService[threadInput.threadId] + if (threadLookupResult.postItemList.isEmpty()) { + val message = "Nothing found for " + threadInput.link + eventRepository.update(logEvent.copy(status = ERROR, message = message)) + return + } +// if (threadItemList.size == 1) { +// threadPoolService.generalExecutor.submit( +// SinglePostRunnable( +// threadItemList[0].postId, threadItemList[0].threadId +// ) +// ) +// eventRepository.update( +// logEvent.copy( +// status = DONE, +// message = "Thread ${thread.link} is added to the queue" +// ) +// ) +// } else { + if (threadRepository.findByThreadId(threadInput.threadId).isEmpty) { + dataTransaction.save( + Thread( + link = threadInput.link, + threadId = threadInput.threadId, + total = threadLookupResult.postItemList.size + ) + ) + eventRepository.update( + logEvent.copy( + status = DONE, message = "New thread ${threadInput.link} is added" + ) + ) + autostart(threadLookupResult) + } else { + log.info("Link ${threadInput.link} is already loaded") + eventRepository.update( + logEvent.copy( + status = ERROR, message = "${threadInput.link} has already been added to the queue" + ) + ) + } +// } + } catch (e: Exception) { + val error = "Error when adding multi-post link ${threadInput.link}" + log.error(error, e) + eventRepository.update( + logEvent.copy( + status = ERROR, message = """ + $error + ${e.formatToString()} + """.trimIndent() + ) + ) + } + } + + private fun autostart(lookupResult: ThreadItem) { + if(lookupResult.postItemList.size <= settings.downloadSettings.autoQueueThreshold) { + downloadService.download(lookupResult.postItemList.map { Pair(it.threadId, it.postId) }) + } + } +} \ No newline at end of file diff --git a/vripper-server/src/main/resources/application.properties b/vripper-core/src/main/resources/application.properties similarity index 95% rename from vripper-server/src/main/resources/application.properties rename to vripper-core/src/main/resources/application.properties index 45ed67d2..e5304200 100644 --- a/vripper-server/src/main/resources/application.properties +++ b/vripper-core/src/main/resources/application.properties @@ -1,6 +1,5 @@ logging.level.org.springframework.web=INFO logging.level.root=INFO -logging.level.org.apache.http=INFO logging.level.org.springframework.web.socket.config.WebSocketMessageBrokerStats=ERROR logging.file.name=${base.dir}/${base.dir.name}/vripper.log server.port=${vripper.server.port:8080} @@ -14,3 +13,4 @@ spring.datasource.password=lEtmEIn spring.datasource.driver-class-name=org.hsqldb.jdbc.JDBCDriver base.dir=${user.dir} base.dir.name=${vripper.base.dir.name:vripper} +download.pool-size=12 \ No newline at end of file diff --git a/vripper-core/src/main/resources/db/changelog/db.changelog-master.xml b/vripper-core/src/main/resources/db/changelog/db.changelog-master.xml new file mode 100644 index 00000000..9de31ef7 --- /dev/null +++ b/vripper-core/src/main/resources/db/changelog/db.changelog-master.xml @@ -0,0 +1,145 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/vripper-core/src/main/resources/logback-spring.xml b/vripper-core/src/main/resources/logback-spring.xml new file mode 100644 index 00000000..8ae0c47c --- /dev/null +++ b/vripper-core/src/main/resources/logback-spring.xml @@ -0,0 +1,24 @@ + + + + + + + + + + + + + + + + + + + + + diff --git a/vripper-server/src/main/resources/proxies.json b/vripper-core/src/main/resources/proxies.json similarity index 100% rename from vripper-server/src/main/resources/proxies.json rename to vripper-core/src/main/resources/proxies.json diff --git a/vripper-electron/.gitignore b/vripper-electron/.gitignore deleted file mode 100644 index c55b0c95..00000000 --- a/vripper-electron/.gitignore +++ /dev/null @@ -1,47 +0,0 @@ -# See http://help.github.com/ignore-files/ for more about ignoring files. - -# compiled output -/dist -/tmp -/out-tsc - -# dependencies -/node_modules - -# profiling files -chrome-profiler-events.json -speed-measure-plugin.json - -# IDEs and editors -/.idea -.project -.classpath -.c9/ -*.launch -.settings/ -*.sublime-workspace - -# IDE - VSCode -.vscode/* -!.vscode/settings.json -!.vscode/tasks.json -!.vscode/launch.json -!.vscode/extensions.json -.history/* - -# misc -/.sass-cache -/connect.lock -/coverage -/libpeerconnection.log -npm-debug.log -yarn-error.log -testem.log -/typings - -# System Files -.DS_Store -Thumbs.db - -dist -build/vripper-ui \ No newline at end of file diff --git a/vripper-electron/.jshintrc b/vripper-electron/.jshintrc deleted file mode 100644 index 1814afde..00000000 --- a/vripper-electron/.jshintrc +++ /dev/null @@ -1,7 +0,0 @@ -{ - "esversion": 6, - "node": true, - "globals": { - "BigInt": false - } -} diff --git a/vripper-electron/build/icon.icns b/vripper-electron/build/icon.icns deleted file mode 100644 index 52f65910..00000000 Binary files a/vripper-electron/build/icon.icns and /dev/null differ diff --git a/vripper-electron/build/icon.ico b/vripper-electron/build/icon.ico deleted file mode 100644 index 9a7584c0..00000000 Binary files a/vripper-electron/build/icon.ico and /dev/null differ diff --git a/vripper-electron/build/icon.png b/vripper-electron/build/icon.png deleted file mode 100644 index 8c1e926d..00000000 Binary files a/vripper-electron/build/icon.png and /dev/null differ diff --git a/vripper-electron/build/icons/1024x1024.png b/vripper-electron/build/icons/1024x1024.png deleted file mode 100644 index 9177aac7..00000000 Binary files a/vripper-electron/build/icons/1024x1024.png and /dev/null differ diff --git a/vripper-electron/build/icons/128x128.png b/vripper-electron/build/icons/128x128.png deleted file mode 100644 index b3627c44..00000000 Binary files a/vripper-electron/build/icons/128x128.png and /dev/null differ diff --git a/vripper-electron/build/icons/16x16.png b/vripper-electron/build/icons/16x16.png deleted file mode 100644 index 7d1d8ac3..00000000 Binary files a/vripper-electron/build/icons/16x16.png and /dev/null differ diff --git a/vripper-electron/build/icons/256x256.png b/vripper-electron/build/icons/256x256.png deleted file mode 100644 index 60708ba4..00000000 Binary files a/vripper-electron/build/icons/256x256.png and /dev/null differ diff --git a/vripper-electron/build/icons/32x32.png b/vripper-electron/build/icons/32x32.png deleted file mode 100644 index 1edd5535..00000000 Binary files a/vripper-electron/build/icons/32x32.png and /dev/null differ diff --git a/vripper-electron/build/icons/48x48.png b/vripper-electron/build/icons/48x48.png deleted file mode 100644 index d2476789..00000000 Binary files a/vripper-electron/build/icons/48x48.png and /dev/null differ diff --git a/vripper-electron/build/icons/512x512.png b/vripper-electron/build/icons/512x512.png deleted file mode 100644 index 5823446a..00000000 Binary files a/vripper-electron/build/icons/512x512.png and /dev/null differ diff --git a/vripper-electron/build/icons/64x64.png b/vripper-electron/build/icons/64x64.png deleted file mode 100644 index 8c1e926d..00000000 Binary files a/vripper-electron/build/icons/64x64.png and /dev/null differ diff --git a/vripper-electron/main.js b/vripper-electron/main.js deleted file mode 100644 index c7866d49..00000000 --- a/vripper-electron/main.js +++ /dev/null @@ -1,182 +0,0 @@ -require('v8-compile-cache'); -const {app, BrowserWindow} = require("electron"); -const path = require("path"); -const url = require("url"); -const getPort = require("get-port"); -const {spawn} = require("child_process"); -const {ipcMain} = require("electron"); -const {dialog} = require("electron"); -const axios = require('axios'); -const windowStateKeeper = require('electron-window-state'); - -// non null value when it is an AppImage -const appImageDir = process.env.APPDIR; -const appImagePath = process.env.APPIMAGE; - -let win; -let vripperServer; -let serverPort; - -const maxTerminationAttemps = 5; -let terminationAttemps = 0; -let terminationInteval; -let terminated = false; - -process.on("uncaughtException", err => { - dialog.showErrorBox(err.message, err.stack); - process.exit(1); -}); - -createWindow = () => { - - if (process.platform === 'win32') { - app.setAppUserModelId('tn.mnlr.vripper'); - } - - let icon; - if (process.platform === 'win32') { - icon = __dirname + '/icon.ico'; - } else if (process.platform === 'linux') { - icon = __dirname + '/icon.png'; - } - - const mainWindowState = windowStateKeeper({ - defaultWidth: 1024, - defaultHeight: 800 - }); - - win = new BrowserWindow({ - x: mainWindowState.x, - y: mainWindowState.y, - width: mainWindowState.width, - height: mainWindowState.height, - minWidth: 800, - minHeight: 600, - webPreferences: { - nodeIntegration: true, - enableRemoteModule: true - }, - icon: icon - }); - - mainWindowState.manage(win); - win.removeMenu(); - win.setMenu(null); - win.setMenuBarVisibility(false); - - win.loadURL( - url.format({ - pathname: path.join(__dirname, `vripper-ui/index.html`), - protocol: "file:", - slashes: true - }) - ); - - // win.webContents.openDevTools(); - - win.on("closed", () => { - win = null; - }); -} - -shutdownServer = () => { - axios.post('http://localhost:' + serverPort + '/actuator/shutdown', {}, { - headers: {'content-type': 'application/json'}, - }).then((response) => { - terminationInteval = setInterval(() => { - terminationAttemps++; - if (terminated) { - console.log('viper server terminated'); - clearInterval(terminationInteval); - app.quit(); - } else if (terminationAttemps > maxTerminationAttemps) { - console.log('viper server is not terminated'); - console.log('Proceed to kill'); - vripperServer.kill('SIGKILL'); - clearInterval(terminationInteval); - app.quit(); - } - }, 1000); - }).catch((error) => { - // Terminate immediately - console.log(error); - vripperServer.kill('SIGKILL'); - terminated = true; - app.quit(); - }); -} - -const gotTheLock = app.requestSingleInstanceLock(); - -if (!gotTheLock) { - app.quit(); -} else { - getPort().then(port => { - serverPort = port; - ipcMain.on("get-port", event => { - event.reply("port", port); - }); - - const appPath = path.join(app.getAppPath(), '../../'); - let javaBinPath, jarPath, baseDir; - if (appImageDir !== undefined && process.platform === 'linux') { - javaBinPath = path.join(appImageDir, "java-runtime/bin/java"); - jarPath = path.join(appImageDir, "bin/vripper-server.jar"); - baseDir = path.join(appImagePath, '..'); - } else if (process.platform === 'darwin') { - javaBinPath = path.join(appPath, "java-runtime/bin/java"); - jarPath = path.join(appPath, "bin/vripper-server.jar"); - baseDir = app.getPath('appData'); - } else if (process.platform === 'win32') { - javaBinPath = path.join(appPath, "java-runtime/bin/java"); - jarPath = path.join(appPath, "bin/vripper-server.jar"); - baseDir = appPath; - } else { - console.error(`Unknown platform ${process.platform}`); - app.quit(); - } - - vripperServer = spawn(javaBinPath, [ - "-Xms256m", - "-Dvripper.server.port=" + port, - "-Dbase.dir=" + baseDir, - "-Djava.net.preferIPv4Stack=true", - "-jar", - jarPath - ], { - stdio: 'ignore' - }); - vripperServer.on('exit', (code, signal) => { - console.log(`vripper server terminated, code = ${code}, signal = ${signal}`); - terminated = true; - app.quit(); - }); - }); - - app.on("second-instance", (event, commandLine, workingDirectory) => { - if (win) { - if (win.isMinimized()) win.restore(); - win.focus(); - } - }); - - app.on("ready", createWindow); - - app.on("window-all-closed", () => { - if (process.platform !== "darwin") { - shutdownServer(); - } - }); - - app.on("will-quit", () => { - if (process.platform === "darwin") { - shutdownServer(); - } - }); - - app.on("activate", () => { - if (win === null) { - createWindow(); - } - }); -} diff --git a/vripper-electron/package.json b/vripper-electron/package.json deleted file mode 100644 index 184f9d27..00000000 --- a/vripper-electron/package.json +++ /dev/null @@ -1,84 +0,0 @@ -{ - "name": "vripper-electron", - "version": "3.5.4", - "description": "A ripper for vipergirls.to built using web technolgies", - "main": "main.js", - "author": "death-claw <53543762+death-claw@users.noreply.github.com>", - "homepage": "https://github.com/death-claw/vripper-project", - "license": "ISC", - "build": { - "appId": "tn.mnlr.vripper", - "productName": "VRipper", - "files": [ - "**/*", - "!**/node_modules/*/{CHANGELOG.md,README.md,README,readme.md,readme}", - "!**/node_modules/*/{test,__tests__,tests,powered-test,example,examples}", - "!**/node_modules/*.d.ts", - "!**/node_modules/.bin", - "!**/*.{iml,o,hprof,orig,pyc,pyo,rbc,swp,csproj,sln,xproj}", - "!.editorconfig", - "!**/._*", - "!**/{.DS_Store,.git,.hg,.svn,CVS,RCS,SCCS,.gitignore,.gitattributes}", - "!**/{__pycache__,thumbs.db,.flowconfig,.idea,.vs,.nyc_output}", - "!**/{appveyor.yml,.travis.yml,circle.yml}", - "!**/{npm-debug.log,yarn.lock,.yarn-integrity,.yarn-metadata.json}", - "!**/build-dir", - "!**/java-runtime", - "!**/pre-build.js", - "!**/pom.xml", - { - "from": "./build/", - "to": "." - } - ], - "extraFiles": [ - { - "from": "../vripper-server/target/vripper-server-${version}-electron.jar", - "to": "bin/vripper-server.jar" - }, - { - "from": "java-runtime", - "to": "java-runtime" - } - ], - "win": { - "icon": "icon.ico", - "target": [ - "dir" - ] - }, - "linux": { - "synopsis": "vipergirls.to ripper", - "category": "Utility", - "packageCategory": "Utility", - "target": [ - "AppImage" - ] - }, - "mac": { - "category": "public.app-category.utilities", - "target": [ - "dmg" - ], - "icon": "icon.icns" - } - }, - "scripts": { - "start": "electron .", - "dist": "node pre-build.js && electron-builder" - }, - "devDependencies": { - "electron": "11.1.1", - "electron-builder": "22.9.1" - }, - "dependencies": { - "axios": "0.21.1", - "cheerio": "1.0.0-rc.3", - "copy-dir": "1.3.0", - "electron-context-menu": "2.1.0", - "electron-window-state": "^5.0.3", - "get-port": "5.1.1", - "rimraf": "3.0.2", - "v8-compile-cache": "2.1.1" - } -} diff --git a/vripper-electron/pom.xml b/vripper-electron/pom.xml deleted file mode 100644 index d812877d..00000000 --- a/vripper-electron/pom.xml +++ /dev/null @@ -1,77 +0,0 @@ - - - 4.0.0 - - tn.mnlr - vripper - 3.5.4 - - vripper-electron - vripper-electron - - - - maven-jar-plugin - - ${buildClassifier} - - - - - - - electron - - electron - - - - tn.mnlr - vripper-server - ${project.version} - runtime - ${buildClassifier} - - - - - - com.github.eirslett - frontend-maven-plugin - 1.11.2 - - v14.16.0 - v1.22.10 - build-dir - - - - install node and yarn - - install-node-and-yarn - - - - yarn install - - yarn - - - - yarn run build - - yarn - - - run dist - - - - - - - - - diff --git a/vripper-electron/pre-build.js b/vripper-electron/pre-build.js deleted file mode 100644 index 829a5240..00000000 --- a/vripper-electron/pre-build.js +++ /dev/null @@ -1,25 +0,0 @@ -const cheerio = require('cheerio'); -const fs = require('fs'); -const rimraf = require("rimraf"); -const copydir = require("copy-dir"); -const {execSync} = require("child_process"); - -rimraf.sync('./build/vripper-ui'); -copydir.sync('../vripper-ui/dist/vripper-ui', './build/vripper-ui', {}); - - -const index = fs.readFileSync('./build/vripper-ui/index.html'); - - -const $ = cheerio.load(index); -$('body').prepend(``); - -fs.writeFileSync('./build/vripper-ui/index.html', $.html()); - -console.log('Building runtime environment'); -rimraf.sync('java-runtime'); -execSync('jlink --no-header-files --no-man-pages --compress=2 --strip-debug --add-modules java.base,java.desktop,java.instrument,java.management,java.security.jgss,java.sql,jdk.unsupported,jdk.crypto.ec --output java-runtime'); -if (process.platform === 'linux') { - console.log('Stripping libjvm.so'); - execSync('strip -p --strip-unneeded java-runtime/lib/server/libjvm.so'); -} diff --git a/vripper-electron/renderer.js b/vripper-electron/renderer.js deleted file mode 100644 index d4001875..00000000 --- a/vripper-electron/renderer.js +++ /dev/null @@ -1,3 +0,0 @@ -console.log("renderer intialized"); -const contextMenu = require("electron-context-menu"); -contextMenu({}); \ No newline at end of file diff --git a/vripper-electron/vripper-electron.iml b/vripper-electron/vripper-electron.iml deleted file mode 100644 index f409c0ea..00000000 --- a/vripper-electron/vripper-electron.iml +++ /dev/null @@ -1,12 +0,0 @@ - - - - - - - - - - - - \ No newline at end of file diff --git a/vripper-electron/yarn.lock b/vripper-electron/yarn.lock deleted file mode 100644 index ae5b1b68..00000000 --- a/vripper-electron/yarn.lock +++ /dev/null @@ -1,1888 +0,0 @@ -# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. -# yarn lockfile v1 - - -"7zip-bin@~5.0.3": - version "5.0.3" - resolved "https://registry.yarnpkg.com/7zip-bin/-/7zip-bin-5.0.3.tgz" - integrity sha512-GLyWIFBbGvpKPGo55JyRZAo4lVbnBiD52cKlw/0Vt+wnmKvWJkpZvsjVoaIolyBXDeAQKSicRtqFNPem9w0WYA== - -"@develar/schema-utils@~2.6.5": - version "2.6.5" - resolved "https://registry.yarnpkg.com/@develar/schema-utils/-/schema-utils-2.6.5.tgz" - integrity sha512-0cp4PsWQ/9avqTVMCtZ+GirikIA36ikvjtHweU4/j8yLtgObI0+JUPhYFScgwlteveGB1rt3Cm8UhN04XayDig== - dependencies: - ajv "^6.12.0" - ajv-keywords "^3.4.1" - -"@electron/get@^1.0.1": - version "1.12.4" - resolved "https://registry.yarnpkg.com/@electron/get/-/get-1.12.4.tgz" - integrity sha512-6nr9DbJPUR9Xujw6zD3y+rS95TyItEVM0NVjt1EehY2vUWfIgPiIPVHxCvaTS0xr2B+DRxovYVKbuOWqC35kjg== - dependencies: - debug "^4.1.1" - env-paths "^2.2.0" - fs-extra "^8.1.0" - got "^9.6.0" - progress "^2.0.3" - semver "^6.2.0" - sumchecker "^3.0.1" - optionalDependencies: - global-agent "^2.0.2" - global-tunnel-ng "^2.7.1" - -"@sindresorhus/is@^0.14.0": - version "0.14.0" - resolved "https://registry.yarnpkg.com/@sindresorhus/is/-/is-0.14.0.tgz" - integrity sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ== - -"@szmarczak/http-timer@^1.1.2": - version "1.1.2" - resolved "https://registry.yarnpkg.com/@szmarczak/http-timer/-/http-timer-1.1.2.tgz" - integrity sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA== - dependencies: - defer-to-connect "^1.0.1" - -"@types/debug@^4.1.5": - version "4.1.5" - resolved "https://registry.yarnpkg.com/@types/debug/-/debug-4.1.5.tgz" - integrity sha512-Q1y515GcOdTHgagaVFhHnIFQ38ygs/kmxdNpvpou+raI9UO3YZcHDngBSYKQklcKlvA7iuQlmIKbzvmxcOE9CQ== - -"@types/fs-extra@^9.0.1": - version "9.0.8" - resolved "https://registry.yarnpkg.com/@types/fs-extra/-/fs-extra-9.0.8.tgz" - integrity sha512-bnlTVTwq03Na7DpWxFJ1dvnORob+Otb8xHyUqUWhqvz/Ksg8+JXPlR52oeMSZ37YEOa5PyccbgUNutiQdi13TA== - dependencies: - "@types/node" "*" - -"@types/node@*": - version "14.14.34" - resolved "https://registry.yarnpkg.com/@types/node/-/node-14.14.34.tgz" - integrity sha512-dBPaxocOK6UVyvhbnpFIj2W+S+1cBTkHQbFQfeeJhoKFbzYcVUGHvddeWPSucKATb3F0+pgDq0i6ghEaZjsugA== - -"@types/node@^12.0.12": - version "12.20.5" - resolved "https://registry.yarnpkg.com/@types/node/-/node-12.20.5.tgz" - integrity sha512-5Oy7tYZnu3a4pnJ//d4yVvOImExl4Vtwf0D40iKUlU+XlUsyV9iyFWyCFlwy489b72FMAik/EFwRkNLjjOdSPg== - -"@types/yargs-parser@*": - version "20.2.0" - resolved "https://registry.yarnpkg.com/@types/yargs-parser/-/yargs-parser-20.2.0.tgz" - integrity sha512-37RSHht+gzzgYeobbG+KWryeAW8J33Nhr69cjTqSYymXVZEN9NbRYWoYlRtDhHKPVT1FyNKwaTPC1NynKZpzRA== - -"@types/yargs@^15.0.5": - version "15.0.13" - resolved "https://registry.yarnpkg.com/@types/yargs/-/yargs-15.0.13.tgz" - integrity sha512-kQ5JNTrbDv3Rp5X2n/iUu37IJBDU2gsZ5R/g1/KHOOEc5IKfUFjXT6DENPGduh08I/pamwtEq4oul7gUqKTQDQ== - dependencies: - "@types/yargs-parser" "*" - -ajv-keywords@^3.4.1: - version "3.5.2" - resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-3.5.2.tgz" - integrity sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ== - -ajv@^6.12.0: - version "6.12.6" - resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz" - integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g== - dependencies: - fast-deep-equal "^3.1.1" - fast-json-stable-stringify "^2.0.0" - json-schema-traverse "^0.4.1" - uri-js "^4.2.2" - -ansi-align@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/ansi-align/-/ansi-align-3.0.0.tgz" - integrity sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw== - dependencies: - string-width "^3.0.0" - -ansi-regex@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-4.1.0.tgz" - integrity sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg== - -ansi-regex@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.0.tgz" - integrity sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg== - -ansi-styles@^3.2.1: - version "3.2.1" - resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz" - integrity sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA== - dependencies: - color-convert "^1.9.0" - -ansi-styles@^4.0.0, ansi-styles@^4.1.0: - version "4.3.0" - resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz" - integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg== - dependencies: - color-convert "^2.0.1" - -app-builder-bin@3.5.10: - version "3.5.10" - resolved "https://registry.yarnpkg.com/app-builder-bin/-/app-builder-bin-3.5.10.tgz" - integrity sha512-Jd+GW68lR0NeetgZDo47PdWBEPdnD+p0jEa7XaxjRC8u6Oo/wgJsfKUkORRgr2NpkD19IFKN50P6JYy04XHFLQ== - -app-builder-lib@22.9.1: - version "22.9.1" - resolved "https://registry.yarnpkg.com/app-builder-lib/-/app-builder-lib-22.9.1.tgz" - integrity sha512-KfXim/fiNwFW2SKffsjEMdAU7RbbEXn62x5YyXle1b4j9X/wEHW9iwox8De6y0hJdR+/kCC/49lI+VgNwLhV7A== - dependencies: - "7zip-bin" "~5.0.3" - "@develar/schema-utils" "~2.6.5" - async-exit-hook "^2.0.1" - bluebird-lst "^1.0.9" - builder-util "22.9.1" - builder-util-runtime "8.7.2" - chromium-pickle-js "^0.2.0" - debug "^4.3.0" - ejs "^3.1.5" - electron-publish "22.9.1" - fs-extra "^9.0.1" - hosted-git-info "^3.0.5" - is-ci "^2.0.0" - isbinaryfile "^4.0.6" - js-yaml "^3.14.0" - lazy-val "^1.0.4" - minimatch "^3.0.4" - normalize-package-data "^2.5.0" - read-config-file "6.0.0" - sanitize-filename "^1.6.3" - semver "^7.3.2" - temp-file "^3.3.7" - -argparse@^1.0.7: - version "1.0.10" - resolved "https://registry.yarnpkg.com/argparse/-/argparse-1.0.10.tgz" - integrity sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg== - dependencies: - sprintf-js "~1.0.2" - -astral-regex@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/astral-regex/-/astral-regex-2.0.0.tgz" - integrity sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ== - -async-exit-hook@^2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/async-exit-hook/-/async-exit-hook-2.0.1.tgz" - integrity sha512-NW2cX8m1Q7KPA7a5M2ULQeZ2wR5qI5PAbw5L0UOMxdioVk9PMZ0h1TmyZEkPYrCvYjDlFICusOu1dlEKAAeXBw== - -async@0.9.x: - version "0.9.2" - resolved "https://registry.yarnpkg.com/async/-/async-0.9.2.tgz" - integrity sha1-rqdNXmHB+JlhO/ZL2mbUx48v0X0= - -at-least-node@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/at-least-node/-/at-least-node-1.0.0.tgz" - integrity sha512-+q/t7Ekv1EDY2l6Gda6LLiX14rU9TV20Wa3ofeQmwPFZbOMo9DXrLbOjFaaclkXKWidIaopwAObQDqwWtGUjqg== - -axios@0.21.1: - version "0.21.1" - resolved "https://registry.yarnpkg.com/axios/-/axios-0.21.1.tgz" - integrity sha512-dKQiRHxGD9PPRIUNIWvZhPTPpl1rf/OxTYKsqKUDjBwYylTvV7SjSHJb9ratfyzM6wCdLCOYLzs73qpg5c4iGA== - dependencies: - follow-redirects "^1.10.0" - -balanced-match@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.0.tgz" - integrity sha1-ibTRmasr7kneFk6gK4nORi1xt2c= - -bluebird-lst@^1.0.9: - version "1.0.9" - resolved "https://registry.yarnpkg.com/bluebird-lst/-/bluebird-lst-1.0.9.tgz" - integrity sha512-7B1Rtx82hjnSD4PGLAjVWeYH3tHAcVUmChh85a3lltKQm6FresXh9ErQo6oAv6CqxttczC3/kEg8SY5NluPuUw== - dependencies: - bluebird "^3.5.5" - -bluebird@^3.5.5: - version "3.7.2" - resolved "https://registry.yarnpkg.com/bluebird/-/bluebird-3.7.2.tgz" - integrity sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg== - -boolbase@~1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/boolbase/-/boolbase-1.0.0.tgz" - integrity sha1-aN/1++YMUes3cl6p4+0xDcwed24= - -boolean@^3.0.1: - version "3.0.2" - resolved "https://registry.yarnpkg.com/boolean/-/boolean-3.0.2.tgz" - integrity sha512-RwywHlpCRc3/Wh81MiCKun4ydaIFyW5Ea6JbL6sRCVx5q5irDw7pMXBUFYF/jArQ6YrG36q0kpovc9P/Kd3I4g== - -boxen@^4.2.0: - version "4.2.0" - resolved "https://registry.yarnpkg.com/boxen/-/boxen-4.2.0.tgz" - integrity sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ== - dependencies: - ansi-align "^3.0.0" - camelcase "^5.3.1" - chalk "^3.0.0" - cli-boxes "^2.2.0" - string-width "^4.1.0" - term-size "^2.1.0" - type-fest "^0.8.1" - widest-line "^3.1.0" - -brace-expansion@^1.1.7: - version "1.1.11" - resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz" - integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA== - dependencies: - balanced-match "^1.0.0" - concat-map "0.0.1" - -buffer-crc32@~0.2.3: - version "0.2.13" - resolved "https://registry.yarnpkg.com/buffer-crc32/-/buffer-crc32-0.2.13.tgz" - integrity sha1-DTM+PwDqxQqhRUq9MO+MKl2ackI= - -buffer-from@^1.0.0: - version "1.1.1" - resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.1.tgz" - integrity sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A== - -builder-util-runtime@8.7.2: - version "8.7.2" - resolved "https://registry.yarnpkg.com/builder-util-runtime/-/builder-util-runtime-8.7.2.tgz" - integrity sha512-xBqv+8bg6cfnzAQK1k3OGpfaHg+QkPgIgpEkXNhouZ0WiUkyZCftuRc2LYzQrLucFywpa14Xbc6+hTbpq83yRA== - dependencies: - debug "^4.1.1" - sax "^1.2.4" - -builder-util@22.9.1: - version "22.9.1" - resolved "https://registry.yarnpkg.com/builder-util/-/builder-util-22.9.1.tgz" - integrity sha512-5hN/XOaYu4ZQUS6F+5CXE6jTo+NAnVqAxDuKGSaHWb9bejfv/rluChTLoY3/nJh7RFjkoyVjvFJv7zQDB1QmHw== - dependencies: - "7zip-bin" "~5.0.3" - "@types/debug" "^4.1.5" - "@types/fs-extra" "^9.0.1" - app-builder-bin "3.5.10" - bluebird-lst "^1.0.9" - builder-util-runtime "8.7.2" - chalk "^4.1.0" - debug "^4.3.0" - fs-extra "^9.0.1" - is-ci "^2.0.0" - js-yaml "^3.14.0" - source-map-support "^0.5.19" - stat-mode "^1.0.0" - temp-file "^3.3.7" - -cacheable-request@^6.0.0: - version "6.1.0" - resolved "https://registry.yarnpkg.com/cacheable-request/-/cacheable-request-6.1.0.tgz" - integrity sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg== - dependencies: - clone-response "^1.0.2" - get-stream "^5.1.0" - http-cache-semantics "^4.0.0" - keyv "^3.0.0" - lowercase-keys "^2.0.0" - normalize-url "^4.1.0" - responselike "^1.0.2" - -camelcase@^5.3.1: - version "5.3.1" - resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-5.3.1.tgz" - integrity sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg== - -chalk@^2.4.2: - version "2.4.2" - resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz" - integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ== - dependencies: - ansi-styles "^3.2.1" - escape-string-regexp "^1.0.5" - supports-color "^5.3.0" - -chalk@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/chalk/-/chalk-3.0.0.tgz" - integrity sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg== - dependencies: - ansi-styles "^4.1.0" - supports-color "^7.1.0" - -chalk@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.0.tgz" - integrity sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A== - dependencies: - ansi-styles "^4.1.0" - supports-color "^7.1.0" - -cheerio@1.0.0-rc.3: - version "1.0.0-rc.3" - resolved "https://registry.yarnpkg.com/cheerio/-/cheerio-1.0.0-rc.3.tgz" - integrity sha512-0td5ijfUPuubwLUu0OBoe98gZj8C/AA+RW3v67GPlGOrvxWjZmBXiBCRU+I8VEiNyJzjth40POfHiz2RB3gImA== - dependencies: - css-select "~1.2.0" - dom-serializer "~0.1.1" - entities "~1.1.1" - htmlparser2 "^3.9.1" - lodash "^4.15.0" - parse5 "^3.0.1" - -chromium-pickle-js@^0.2.0: - version "0.2.0" - resolved "https://registry.yarnpkg.com/chromium-pickle-js/-/chromium-pickle-js-0.2.0.tgz" - integrity sha1-BKEGZywYsIWrd02YPfo+oTjyIgU= - -ci-info@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/ci-info/-/ci-info-2.0.0.tgz" - integrity sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ== - -cli-boxes@^2.2.0: - version "2.2.1" - resolved "https://registry.yarnpkg.com/cli-boxes/-/cli-boxes-2.2.1.tgz" - integrity sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw== - -cli-truncate@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/cli-truncate/-/cli-truncate-2.1.0.tgz" - integrity sha512-n8fOixwDD6b/ObinzTrp1ZKFzbgvKZvuz/TvejnLn1aQfC6r52XEx85FmuC+3HI+JM7coBRXUvNqEU2PHVrHpg== - dependencies: - slice-ansi "^3.0.0" - string-width "^4.2.0" - -cliui@^7.0.2: - version "7.0.4" - resolved "https://registry.yarnpkg.com/cliui/-/cliui-7.0.4.tgz" - integrity sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ== - dependencies: - string-width "^4.2.0" - strip-ansi "^6.0.0" - wrap-ansi "^7.0.0" - -clone-response@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/clone-response/-/clone-response-1.0.2.tgz" - integrity sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws= - dependencies: - mimic-response "^1.0.0" - -color-convert@^1.9.0: - version "1.9.3" - resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.3.tgz" - integrity sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg== - dependencies: - color-name "1.1.3" - -color-convert@^2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz" - integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ== - dependencies: - color-name "~1.1.4" - -color-name@1.1.3: - version "1.1.3" - resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz" - integrity sha1-p9BVi9icQveV3UIyj3QIMcpTvCU= - -color-name@~1.1.4: - version "1.1.4" - resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz" - integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== - -concat-map@0.0.1: - version "0.0.1" - resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz" - integrity sha1-2Klr13/Wjfd5OnMDajug1UBdR3s= - -concat-stream@^1.6.2: - version "1.6.2" - resolved "https://registry.yarnpkg.com/concat-stream/-/concat-stream-1.6.2.tgz" - integrity sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw== - dependencies: - buffer-from "^1.0.0" - inherits "^2.0.3" - readable-stream "^2.2.2" - typedarray "^0.0.6" - -config-chain@^1.1.11: - version "1.1.12" - resolved "https://registry.yarnpkg.com/config-chain/-/config-chain-1.1.12.tgz" - integrity sha512-a1eOIcu8+7lUInge4Rpf/n4Krkf3Dd9lqhljRzII1/Zno/kRtUWnznPO3jOKBmTEktkt3fkxisUcivoj0ebzoA== - dependencies: - ini "^1.3.4" - proto-list "~1.2.1" - -configstore@^5.0.1: - version "5.0.1" - resolved "https://registry.yarnpkg.com/configstore/-/configstore-5.0.1.tgz" - integrity sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA== - dependencies: - dot-prop "^5.2.0" - graceful-fs "^4.1.2" - make-dir "^3.0.0" - unique-string "^2.0.0" - write-file-atomic "^3.0.0" - xdg-basedir "^4.0.0" - -copy-dir@1.3.0: - version "1.3.0" - resolved "https://registry.yarnpkg.com/copy-dir/-/copy-dir-1.3.0.tgz" - integrity sha512-Q4+qBFnN4bwGwvtXXzbp4P/4iNk0MaiGAzvQ8OiMtlLjkIKjmNN689uVzShSM0908q7GoFHXIPx4zi75ocoaHw== - -core-js@^3.6.5: - version "3.9.1" - resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.9.1.tgz" - integrity sha512-gSjRvzkxQc1zjM/5paAmL4idJBFzuJoo+jDjF1tStYFMV2ERfD02HhahhCGXUyHxQRG4yFKVSdO6g62eoRMcDg== - -core-util-is@~1.0.0: - version "1.0.2" - resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz" - integrity sha1-tf1UIgqivFq1eqtxQMlAdUUDwac= - -crypto-random-string@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/crypto-random-string/-/crypto-random-string-2.0.0.tgz" - integrity sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA== - -css-select@~1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/css-select/-/css-select-1.2.0.tgz" - integrity sha1-KzoRBTnFNV8c2NMUYj6HCxIeyFg= - dependencies: - boolbase "~1.0.0" - css-what "2.1" - domutils "1.5.1" - nth-check "~1.0.1" - -css-what@2.1: - version "2.1.3" - resolved "https://registry.yarnpkg.com/css-what/-/css-what-2.1.3.tgz" - integrity sha512-a+EPoD+uZiNfh+5fxw2nO9QwFa6nJe2Or35fGY6Ipw1R3R4AGz1d1TEZrCegvw2YTmZ0jXirGYlzxxpYSHwpEg== - -debug@^2.6.9: - version "2.6.9" - resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz" - integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA== - dependencies: - ms "2.0.0" - -debug@^4.1.0, debug@^4.1.1, debug@^4.3.0: - version "4.3.1" - resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.1.tgz" - integrity sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ== - dependencies: - ms "2.1.2" - -decompress-response@^3.3.0: - version "3.3.0" - resolved "https://registry.yarnpkg.com/decompress-response/-/decompress-response-3.3.0.tgz" - integrity sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M= - dependencies: - mimic-response "^1.0.0" - -deep-extend@^0.6.0: - version "0.6.0" - resolved "https://registry.yarnpkg.com/deep-extend/-/deep-extend-0.6.0.tgz" - integrity sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA== - -defer-to-connect@^1.0.1: - version "1.1.3" - resolved "https://registry.yarnpkg.com/defer-to-connect/-/defer-to-connect-1.1.3.tgz" - integrity sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ== - -define-properties@^1.1.3: - version "1.1.3" - resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.1.3.tgz" - integrity sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ== - dependencies: - object-keys "^1.0.12" - -detect-node@^2.0.4: - version "2.0.4" - resolved "https://registry.yarnpkg.com/detect-node/-/detect-node-2.0.4.tgz" - integrity sha512-ZIzRpLJrOj7jjP2miAtgqIfmzbxa4ZOr5jJc601zklsfEx9oTzmmj2nVpIPRpNlRTIh8lc1kyViIY7BWSGNmKw== - -dmg-builder@22.9.1: - version "22.9.1" - resolved "https://registry.yarnpkg.com/dmg-builder/-/dmg-builder-22.9.1.tgz" - integrity sha512-jc+DAirqmQrNT6KbDHdfEp8D1kD0DBTnsLhwUR3MX+hMBun5bT134LQzpdK0GKvd22GqF8L1Cz/NOgaVjscAXQ== - dependencies: - app-builder-lib "22.9.1" - builder-util "22.9.1" - fs-extra "^9.0.1" - iconv-lite "^0.6.2" - js-yaml "^3.14.0" - sanitize-filename "^1.6.3" - -dom-serializer@0: - version "0.2.2" - resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-0.2.2.tgz" - integrity sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g== - dependencies: - domelementtype "^2.0.1" - entities "^2.0.0" - -dom-serializer@~0.1.1: - version "0.1.1" - resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-0.1.1.tgz" - integrity sha512-l0IU0pPzLWSHBcieZbpOKgkIn3ts3vAh7ZuFyXNwJxJXk/c4Gwj9xaTJwIDVQCXawWD0qb3IzMGH5rglQaO0XA== - dependencies: - domelementtype "^1.3.0" - entities "^1.1.1" - -domelementtype@1, domelementtype@^1.3.0, domelementtype@^1.3.1: - version "1.3.1" - resolved "https://registry.yarnpkg.com/domelementtype/-/domelementtype-1.3.1.tgz" - integrity sha512-BSKB+TSpMpFI/HOxCNr1O8aMOTZ8hT3pM3GQ0w/mWRmkhEDSFJkkyzz4XQsBV44BChwGkrDfMyjVD0eA2aFV3w== - -domelementtype@^2.0.1: - version "2.1.0" - resolved "https://registry.yarnpkg.com/domelementtype/-/domelementtype-2.1.0.tgz" - integrity sha512-LsTgx/L5VpD+Q8lmsXSHW2WpA+eBlZ9HPf3erD1IoPF00/3JKHZ3BknUVA2QGDNu69ZNmyFmCWBSO45XjYKC5w== - -domhandler@^2.3.0: - version "2.4.2" - resolved "https://registry.yarnpkg.com/domhandler/-/domhandler-2.4.2.tgz" - integrity sha512-JiK04h0Ht5u/80fdLMCEmV4zkNh2BcoMFBmZ/91WtYZ8qVXSKjiw7fXMgFPnHcSZgOo3XdinHvmnDUeMf5R4wA== - dependencies: - domelementtype "1" - -domutils@1.5.1: - version "1.5.1" - resolved "https://registry.yarnpkg.com/domutils/-/domutils-1.5.1.tgz" - integrity sha1-3NhIiib1Y9YQeeSMn3t+Mjc2gs8= - dependencies: - dom-serializer "0" - domelementtype "1" - -domutils@^1.5.1: - version "1.7.0" - resolved "https://registry.yarnpkg.com/domutils/-/domutils-1.7.0.tgz" - integrity sha512-Lgd2XcJ/NjEw+7tFvfKxOzCYKZsdct5lczQ2ZaQY8Djz7pfAD3Gbp8ySJWtreII/vDlMVmxwa6pHmdxIYgttDg== - dependencies: - dom-serializer "0" - domelementtype "1" - -dot-prop@^5.2.0: - version "5.3.0" - resolved "https://registry.yarnpkg.com/dot-prop/-/dot-prop-5.3.0.tgz" - integrity sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q== - dependencies: - is-obj "^2.0.0" - -dotenv-expand@^5.1.0: - version "5.1.0" - resolved "https://registry.yarnpkg.com/dotenv-expand/-/dotenv-expand-5.1.0.tgz" - integrity sha512-YXQl1DSa4/PQyRfgrv6aoNjhasp/p4qs9FjJ4q4cQk+8m4r6k4ZSiEyytKG8f8W9gi8WsQtIObNmKd+tMzNTmA== - -dotenv@^8.2.0: - version "8.2.0" - resolved "https://registry.yarnpkg.com/dotenv/-/dotenv-8.2.0.tgz" - integrity sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw== - -duplexer3@^0.1.4: - version "0.1.4" - resolved "https://registry.yarnpkg.com/duplexer3/-/duplexer3-0.1.4.tgz" - integrity sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI= - -ejs@^3.1.5: - version "3.1.6" - resolved "https://registry.yarnpkg.com/ejs/-/ejs-3.1.6.tgz" - integrity sha512-9lt9Zse4hPucPkoP7FHDF0LQAlGyF9JVpnClFLFH3aSSbxmyoqINRpp/9wePWJTUl4KOQwRL72Iw3InHPDkoGw== - dependencies: - jake "^10.6.1" - -electron-builder@22.9.1: - version "22.9.1" - resolved "https://registry.yarnpkg.com/electron-builder/-/electron-builder-22.9.1.tgz" - integrity sha512-GXPt8l5Mxwm1QKYopUM6/Tdh9W3695G6Ax+IFyj5pQ51G4SD5L1uq4/RkPSsOgs3rP7jNSV6g6OfDzdtVufPdA== - dependencies: - "@types/yargs" "^15.0.5" - app-builder-lib "22.9.1" - bluebird-lst "^1.0.9" - builder-util "22.9.1" - builder-util-runtime "8.7.2" - chalk "^4.1.0" - dmg-builder "22.9.1" - fs-extra "^9.0.1" - is-ci "^2.0.0" - lazy-val "^1.0.4" - read-config-file "6.0.0" - sanitize-filename "^1.6.3" - update-notifier "^4.1.1" - yargs "^16.0.3" - -electron-context-menu@2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/electron-context-menu/-/electron-context-menu-2.1.0.tgz" - integrity sha512-xnJS4C24W/h6rw2fbRYMiHCHWzbJeuhPTLJR88lJ+jSnO8VKY0oOolE2AeL/zTTiUYrOXF+3POcfpSoaF0d8Pw== - dependencies: - cli-truncate "^2.0.0" - electron-dl "^3.0.0" - electron-is-dev "^1.0.1" - -electron-dl@^3.0.0: - version "3.2.1" - resolved "https://registry.yarnpkg.com/electron-dl/-/electron-dl-3.2.1.tgz" - integrity sha512-k5DFjocJlXbrjshO1zeWe/Gz7HkGwCgnehHPemiyzN2B/LfLlnbIX7sCj5F+huTwZ2l+nQehTI4IR37xvCn6FQ== - dependencies: - ext-name "^5.0.0" - pupa "^2.0.1" - unused-filename "^2.1.0" - -electron-is-dev@^1.0.1: - version "1.2.0" - resolved "https://registry.yarnpkg.com/electron-is-dev/-/electron-is-dev-1.2.0.tgz" - integrity sha512-R1oD5gMBPS7PVU8gJwH6CtT0e6VSoD0+SzSnYpNm+dBkcijgA+K7VAMHDfnRq/lkKPZArpzplTW6jfiMYosdzw== - -electron-publish@22.9.1: - version "22.9.1" - resolved "https://registry.yarnpkg.com/electron-publish/-/electron-publish-22.9.1.tgz" - integrity sha512-ducLjRJLEeU87FaTCWaUyDjCoLXHkawkltP2zqS/n2PyGke54ZIql0tBuUheht4EpR8AhFbVJ11spSn1gy8r6w== - dependencies: - "@types/fs-extra" "^9.0.1" - bluebird-lst "^1.0.9" - builder-util "22.9.1" - builder-util-runtime "8.7.2" - chalk "^4.1.0" - fs-extra "^9.0.1" - lazy-val "^1.0.4" - mime "^2.4.6" - -electron-window-state@^5.0.3: - version "5.0.3" - resolved "https://registry.npmjs.org/electron-window-state/-/electron-window-state-5.0.3.tgz" - integrity sha512-1mNTwCfkolXl3kMf50yW3vE2lZj0y92P/HYWFBrb+v2S/pCka5mdwN3cagKm458A7NjndSwijynXgcLWRodsVg== - dependencies: - jsonfile "^4.0.0" - mkdirp "^0.5.1" - -electron@11.1.1: - version "11.1.1" - resolved "https://registry.yarnpkg.com/electron/-/electron-11.1.1.tgz" - integrity sha512-tlbex3xosJgfileN6BAQRotevPRXB/wQIq48QeQ08tUJJrXwE72c8smsM/hbHx5eDgnbfJ2G3a60PmRjHU2NhA== - dependencies: - "@electron/get" "^1.0.1" - "@types/node" "^12.0.12" - extract-zip "^1.0.3" - -emoji-regex@^7.0.1: - version "7.0.3" - resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-7.0.3.tgz" - integrity sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA== - -emoji-regex@^8.0.0: - version "8.0.0" - resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz" - integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A== - -encodeurl@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz" - integrity sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k= - -end-of-stream@^1.1.0: - version "1.4.4" - resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.4.tgz" - integrity sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q== - dependencies: - once "^1.4.0" - -entities@^1.1.1, entities@~1.1.1: - version "1.1.2" - resolved "https://registry.yarnpkg.com/entities/-/entities-1.1.2.tgz" - integrity sha512-f2LZMYl1Fzu7YSBKg+RoROelpOaNrcGmE9AZubeDfrCEia483oW4MI4VyFd5VNHIgQ/7qm1I0wUHK1eJnn2y2w== - -entities@^2.0.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/entities/-/entities-2.2.0.tgz" - integrity sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A== - -env-paths@^2.2.0: - version "2.2.1" - resolved "https://registry.yarnpkg.com/env-paths/-/env-paths-2.2.1.tgz" - integrity sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A== - -es6-error@^4.1.1: - version "4.1.1" - resolved "https://registry.yarnpkg.com/es6-error/-/es6-error-4.1.1.tgz" - integrity sha512-Um/+FxMr9CISWh0bi5Zv0iOD+4cFh5qLeks1qhAopKVAJw3drgKbKySikp7wGhDL0HPeaja0P5ULZrxLkniUVg== - -escalade@^3.1.1: - version "3.1.1" - resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.1.1.tgz" - integrity sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw== - -escape-goat@^2.0.0: - version "2.1.1" - resolved "https://registry.yarnpkg.com/escape-goat/-/escape-goat-2.1.1.tgz" - integrity sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q== - -escape-string-regexp@^1.0.5: - version "1.0.5" - resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz" - integrity sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ= - -escape-string-regexp@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz" - integrity sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA== - -esprima@^4.0.0: - version "4.0.1" - resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz" - integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A== - -ext-list@^2.0.0: - version "2.2.2" - resolved "https://registry.yarnpkg.com/ext-list/-/ext-list-2.2.2.tgz" - integrity sha512-u+SQgsubraE6zItfVA0tBuCBhfU9ogSRnsvygI7wht9TS510oLkBRXBsqopeUG/GBOIQyKZO9wjTqIu/sf5zFA== - dependencies: - mime-db "^1.28.0" - -ext-name@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/ext-name/-/ext-name-5.0.0.tgz" - integrity sha512-yblEwXAbGv1VQDmow7s38W77hzAgJAO50ztBLMcUyUBfxv1HC+LGwtiEN+Co6LtlqT/5uwVOxsD4TNIilWhwdQ== - dependencies: - ext-list "^2.0.0" - sort-keys-length "^1.0.0" - -extract-zip@^1.0.3: - version "1.7.0" - resolved "https://registry.yarnpkg.com/extract-zip/-/extract-zip-1.7.0.tgz" - integrity sha512-xoh5G1W/PB0/27lXgMQyIhP5DSY/LhoCsOyZgb+6iMmRtCwVBo55uKaMoEYrDCKQhWvqEip5ZPKAc6eFNyf/MA== - dependencies: - concat-stream "^1.6.2" - debug "^2.6.9" - mkdirp "^0.5.4" - yauzl "^2.10.0" - -fast-deep-equal@^3.1.1: - version "3.1.3" - resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz" - integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q== - -fast-json-stable-stringify@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz" - integrity sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw== - -fd-slicer@~1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/fd-slicer/-/fd-slicer-1.1.0.tgz" - integrity sha1-JcfInLH5B3+IkbvmHY85Dq4lbx4= - dependencies: - pend "~1.2.0" - -filelist@^1.0.1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/filelist/-/filelist-1.0.2.tgz" - integrity sha512-z7O0IS8Plc39rTCq6i6iHxk43duYOn8uFJiWSewIq0Bww1RNybVHSCjahmcC87ZqAm4OTvFzlzeGu3XAzG1ctQ== - dependencies: - minimatch "^3.0.4" - -follow-redirects@^1.10.0: - version "1.13.3" - resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.13.3.tgz" - integrity sha512-DUgl6+HDzB0iEptNQEXLx/KhTmDb8tZUHSeLqpnjpknR70H0nC2t9N73BK6fN4hOvJ84pKlIQVQ4k5FFlBedKA== - -fs-extra@^8.1.0: - version "8.1.0" - resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-8.1.0.tgz" - integrity sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g== - dependencies: - graceful-fs "^4.2.0" - jsonfile "^4.0.0" - universalify "^0.1.0" - -fs-extra@^9.0.1: - version "9.1.0" - resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-9.1.0.tgz" - integrity sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ== - dependencies: - at-least-node "^1.0.0" - graceful-fs "^4.2.0" - jsonfile "^6.0.1" - universalify "^2.0.0" - -fs.realpath@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz" - integrity sha1-FQStJSMVjKpA20onh8sBQRmU6k8= - -function-bind@^1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.1.tgz" - integrity sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A== - -get-caller-file@^2.0.5: - version "2.0.5" - resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz" - integrity sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg== - -get-port@5.1.1: - version "5.1.1" - resolved "https://registry.yarnpkg.com/get-port/-/get-port-5.1.1.tgz" - integrity sha512-g/Q1aTSDOxFpchXC4i8ZWvxA1lnPqx/JHqcpIw0/LX9T8x/GBbi6YnlN5nhaKIFkT8oFsscUKgDJYxfwfS6QsQ== - -get-stream@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-4.1.0.tgz" - integrity sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w== - dependencies: - pump "^3.0.0" - -get-stream@^5.1.0: - version "5.2.0" - resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-5.2.0.tgz" - integrity sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA== - dependencies: - pump "^3.0.0" - -glob@^7.1.3: - version "7.1.6" - resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.6.tgz" - integrity sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA== - dependencies: - fs.realpath "^1.0.0" - inflight "^1.0.4" - inherits "2" - minimatch "^3.0.4" - once "^1.3.0" - path-is-absolute "^1.0.0" - -global-agent@^2.0.2: - version "2.1.12" - resolved "https://registry.yarnpkg.com/global-agent/-/global-agent-2.1.12.tgz" - integrity sha512-caAljRMS/qcDo69X9BfkgrihGUgGx44Fb4QQToNQjsiWh+YlQ66uqYVAdA8Olqit+5Ng0nkz09je3ZzANMZcjg== - dependencies: - boolean "^3.0.1" - core-js "^3.6.5" - es6-error "^4.1.1" - matcher "^3.0.0" - roarr "^2.15.3" - semver "^7.3.2" - serialize-error "^7.0.1" - -global-dirs@^2.0.1: - version "2.1.0" - resolved "https://registry.yarnpkg.com/global-dirs/-/global-dirs-2.1.0.tgz" - integrity sha512-MG6kdOUh/xBnyo9cJFeIKkLEc1AyFq42QTU4XiX51i2NEdxLxLWXIjEjmqKeSuKR7pAZjTqUVoT2b2huxVLgYQ== - dependencies: - ini "1.3.7" - -global-tunnel-ng@^2.7.1: - version "2.7.1" - resolved "https://registry.yarnpkg.com/global-tunnel-ng/-/global-tunnel-ng-2.7.1.tgz" - integrity sha512-4s+DyciWBV0eK148wqXxcmVAbFVPqtc3sEtUE/GTQfuU80rySLcMhUmHKSHI7/LDj8q0gDYI1lIhRRB7ieRAqg== - dependencies: - encodeurl "^1.0.2" - lodash "^4.17.10" - npm-conf "^1.1.3" - tunnel "^0.0.6" - -globalthis@^1.0.1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/globalthis/-/globalthis-1.0.2.tgz" - integrity sha512-ZQnSFO1la8P7auIOQECnm0sSuoMeaSq0EEdXMBFF2QJO4uNcwbyhSgG3MruWNbFTqCLmxVwGOl7LZ9kASvHdeQ== - dependencies: - define-properties "^1.1.3" - -got@^9.6.0: - version "9.6.0" - resolved "https://registry.yarnpkg.com/got/-/got-9.6.0.tgz" - integrity sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q== - dependencies: - "@sindresorhus/is" "^0.14.0" - "@szmarczak/http-timer" "^1.1.2" - cacheable-request "^6.0.0" - decompress-response "^3.3.0" - duplexer3 "^0.1.4" - get-stream "^4.1.0" - lowercase-keys "^1.0.1" - mimic-response "^1.0.1" - p-cancelable "^1.0.0" - to-readable-stream "^1.0.0" - url-parse-lax "^3.0.0" - -graceful-fs@^4.1.2, graceful-fs@^4.1.6, graceful-fs@^4.2.0: - version "4.2.6" - resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.6.tgz" - integrity sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ== - -has-flag@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz" - integrity sha1-tdRU3CGZriJWmfNGfloH87lVuv0= - -has-flag@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz" - integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ== - -has-yarn@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/has-yarn/-/has-yarn-2.1.0.tgz" - integrity sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw== - -has@^1.0.3: - version "1.0.3" - resolved "https://registry.yarnpkg.com/has/-/has-1.0.3.tgz" - integrity sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw== - dependencies: - function-bind "^1.1.1" - -hosted-git-info@^2.1.4: - version "2.8.9" - resolved "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz" - integrity sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw== - -hosted-git-info@^3.0.5: - version "3.0.8" - resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-3.0.8.tgz" - integrity sha512-aXpmwoOhRBrw6X3j0h5RloK4x1OzsxMPyxqIHyNfSe2pypkVTZFpEiRoSipPEPlMrh0HW/XsjkJ5WgnCirpNUw== - dependencies: - lru-cache "^6.0.0" - -htmlparser2@^3.9.1: - version "3.10.1" - resolved "https://registry.yarnpkg.com/htmlparser2/-/htmlparser2-3.10.1.tgz" - integrity sha512-IgieNijUMbkDovyoKObU1DUhm1iwNYE/fuifEoEHfd1oZKZDaONBSkal7Y01shxsM49R4XaMdGez3WnF9UfiCQ== - dependencies: - domelementtype "^1.3.1" - domhandler "^2.3.0" - domutils "^1.5.1" - entities "^1.1.1" - inherits "^2.0.1" - readable-stream "^3.1.1" - -http-cache-semantics@^4.0.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz" - integrity sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ== - -iconv-lite@^0.6.2: - version "0.6.2" - resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.6.2.tgz" - integrity sha512-2y91h5OpQlolefMPmUlivelittSWy0rP+oYVpn6A7GwVHNE8AWzoYOBNmlwks3LobaJxgHCYZAnyNo2GgpNRNQ== - dependencies: - safer-buffer ">= 2.1.2 < 3.0.0" - -import-lazy@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/import-lazy/-/import-lazy-2.1.0.tgz" - integrity sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM= - -imurmurhash@^0.1.4: - version "0.1.4" - resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz" - integrity sha1-khi5srkoojixPcT7a21XbyMUU+o= - -inflight@^1.0.4: - version "1.0.6" - resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz" - integrity sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk= - dependencies: - once "^1.3.0" - wrappy "1" - -inherits@2, inherits@^2.0.1, inherits@^2.0.3, inherits@~2.0.3: - version "2.0.4" - resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz" - integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== - -ini@1.3.7: - version "1.3.7" - resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.7.tgz" - integrity sha512-iKpRpXP+CrP2jyrxvg1kMUpXDyRUFDWurxbnVT1vQPx+Wz9uCYsMIqYuSBLV+PAaZG/d7kRLKRFc9oDMsH+mFQ== - -ini@^1.3.4, ini@~1.3.0: - version "1.3.8" - resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.8.tgz" - integrity sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew== - -is-ci@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/is-ci/-/is-ci-2.0.0.tgz" - integrity sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w== - dependencies: - ci-info "^2.0.0" - -is-core-module@^2.2.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.2.0.tgz" - integrity sha512-XRAfAdyyY5F5cOXn7hYQDqh2Xmii+DEfIcQGxK/uNwMHhIkPWO0g8msXcbzLe+MpGoR951MlqM/2iIlU4vKDdQ== - dependencies: - has "^1.0.3" - -is-fullwidth-code-point@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz" - integrity sha1-o7MKXE8ZkYMWeqq5O+764937ZU8= - -is-fullwidth-code-point@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz" - integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg== - -is-installed-globally@^0.3.1: - version "0.3.2" - resolved "https://registry.yarnpkg.com/is-installed-globally/-/is-installed-globally-0.3.2.tgz" - integrity sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g== - dependencies: - global-dirs "^2.0.1" - is-path-inside "^3.0.1" - -is-npm@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/is-npm/-/is-npm-4.0.0.tgz" - integrity sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig== - -is-obj@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-2.0.0.tgz" - integrity sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w== - -is-path-inside@^3.0.1: - version "3.0.3" - resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-3.0.3.tgz" - integrity sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ== - -is-plain-obj@^1.0.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/is-plain-obj/-/is-plain-obj-1.1.0.tgz" - integrity sha1-caUMhCnfync8kqOQpKA7OfzVHT4= - -is-typedarray@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/is-typedarray/-/is-typedarray-1.0.0.tgz" - integrity sha1-5HnICFjfDBsR3dppQPlgEfzaSpo= - -is-yarn-global@^0.3.0: - version "0.3.0" - resolved "https://registry.yarnpkg.com/is-yarn-global/-/is-yarn-global-0.3.0.tgz" - integrity sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw== - -isarray@~1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz" - integrity sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE= - -isbinaryfile@^4.0.6: - version "4.0.6" - resolved "https://registry.yarnpkg.com/isbinaryfile/-/isbinaryfile-4.0.6.tgz" - integrity sha512-ORrEy+SNVqUhrCaal4hA4fBzhggQQ+BaLntyPOdoEiwlKZW9BZiJXjg3RMiruE4tPEI3pyVPpySHQF/dKWperg== - -jake@^10.6.1: - version "10.8.2" - resolved "https://registry.yarnpkg.com/jake/-/jake-10.8.2.tgz" - integrity sha512-eLpKyrfG3mzvGE2Du8VoPbeSkRry093+tyNjdYaBbJS9v17knImYGNXQCUV0gLxQtF82m3E8iRb/wdSQZLoq7A== - dependencies: - async "0.9.x" - chalk "^2.4.2" - filelist "^1.0.1" - minimatch "^3.0.4" - -js-yaml@^3.13.1, js-yaml@^3.14.0: - version "3.14.1" - resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.14.1.tgz" - integrity sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g== - dependencies: - argparse "^1.0.7" - esprima "^4.0.0" - -json-buffer@3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/json-buffer/-/json-buffer-3.0.0.tgz" - integrity sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg= - -json-schema-traverse@^0.4.1: - version "0.4.1" - resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz" - integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg== - -json-stringify-safe@^5.0.1: - version "5.0.1" - resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz" - integrity sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus= - -json5@^2.1.2: - version "2.2.0" - resolved "https://registry.yarnpkg.com/json5/-/json5-2.2.0.tgz" - integrity sha512-f+8cldu7X/y7RAJurMEJmdoKXGB/X550w2Nr3tTbezL6RwEE/iMcm+tZnXeoZtKuOq6ft8+CqzEkrIgx1fPoQA== - dependencies: - minimist "^1.2.5" - -jsonfile@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-4.0.0.tgz" - integrity sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss= - optionalDependencies: - graceful-fs "^4.1.6" - -jsonfile@^6.0.1: - version "6.1.0" - resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-6.1.0.tgz" - integrity sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ== - dependencies: - universalify "^2.0.0" - optionalDependencies: - graceful-fs "^4.1.6" - -keyv@^3.0.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/keyv/-/keyv-3.1.0.tgz" - integrity sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA== - dependencies: - json-buffer "3.0.0" - -latest-version@^5.0.0: - version "5.1.0" - resolved "https://registry.yarnpkg.com/latest-version/-/latest-version-5.1.0.tgz" - integrity sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA== - dependencies: - package-json "^6.3.0" - -lazy-val@^1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/lazy-val/-/lazy-val-1.0.4.tgz" - integrity sha512-u93kb2fPbIrfzBuLjZE+w+fJbUUMhNDXxNmMfaqNgpfQf1CO5ZSe2LfsnBqVAk7i/2NF48OSoRj+Xe2VT+lE8Q== - -lodash@^4.15.0, lodash@^4.17.10: - version "4.17.21" - resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz" - integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg== - -lowercase-keys@^1.0.0, lowercase-keys@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-1.0.1.tgz" - integrity sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA== - -lowercase-keys@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-2.0.0.tgz" - integrity sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA== - -lru-cache@^6.0.0: - version "6.0.0" - resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz" - integrity sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA== - dependencies: - yallist "^4.0.0" - -make-dir@^3.0.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-3.1.0.tgz" - integrity sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw== - dependencies: - semver "^6.0.0" - -matcher@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/matcher/-/matcher-3.0.0.tgz" - integrity sha512-OkeDaAZ/bQCxeFAozM55PKcKU0yJMPGifLwV4Qgjitu+5MoAfSQN4lsLJeXZ1b8w0x+/Emda6MZgXS1jvsapng== - dependencies: - escape-string-regexp "^4.0.0" - -mime-db@^1.28.0: - version "1.46.0" - resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.46.0.tgz" - integrity sha512-svXaP8UQRZ5K7or+ZmfNhg2xX3yKDMUzqadsSqi4NCH/KomcH75MAMYAGVlvXn4+b/xOPhS3I2uHKRUzvjY7BQ== - -mime@^2.4.6: - version "2.5.2" - resolved "https://registry.yarnpkg.com/mime/-/mime-2.5.2.tgz" - integrity sha512-tqkh47FzKeCPD2PUiPB6pkbMzsCasjxAfC62/Wap5qrUWcb+sFasXUC5I3gYM5iBM8v/Qpn4UK0x+j0iHyFPDg== - -mimic-response@^1.0.0, mimic-response@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-1.0.1.tgz" - integrity sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ== - -minimatch@^3.0.4: - version "3.0.4" - resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.0.4.tgz" - integrity sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA== - dependencies: - brace-expansion "^1.1.7" - -minimist@^1.2.0, minimist@^1.2.5: - version "1.2.5" - resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.5.tgz" - integrity sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw== - -mkdirp@^0.5.1, mkdirp@^0.5.4: - version "0.5.5" - resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.5.tgz" - integrity sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ== - dependencies: - minimist "^1.2.5" - -modify-filename@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/modify-filename/-/modify-filename-1.1.0.tgz" - integrity sha1-mi3sg4Bvuy2XXyK+7IWcoms5OqE= - -ms@2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz" - integrity sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g= - -ms@2.1.2: - version "2.1.2" - resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz" - integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w== - -normalize-package-data@^2.5.0: - version "2.5.0" - resolved "https://registry.yarnpkg.com/normalize-package-data/-/normalize-package-data-2.5.0.tgz" - integrity sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA== - dependencies: - hosted-git-info "^2.1.4" - resolve "^1.10.0" - semver "2 || 3 || 4 || 5" - validate-npm-package-license "^3.0.1" - -normalize-url@^4.1.0: - version "4.5.0" - resolved "https://registry.yarnpkg.com/normalize-url/-/normalize-url-4.5.0.tgz" - integrity sha512-2s47yzUxdexf1OhyRi4Em83iQk0aPvwTddtFz4hnSSw9dCEsLEGf6SwIO8ss/19S9iBb5sJaOuTvTGDeZI00BQ== - -npm-conf@^1.1.3: - version "1.1.3" - resolved "https://registry.yarnpkg.com/npm-conf/-/npm-conf-1.1.3.tgz" - integrity sha512-Yic4bZHJOt9RCFbRP3GgpqhScOY4HH3V2P8yBj6CeYq118Qr+BLXqT2JvpJ00mryLESpgOxf5XlFv4ZjXxLScw== - dependencies: - config-chain "^1.1.11" - pify "^3.0.0" - -nth-check@~1.0.1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/nth-check/-/nth-check-1.0.2.tgz" - integrity sha512-WeBOdju8SnzPN5vTUJYxYUxLeXpCaVP5i5e0LF8fg7WORF2Wd7wFX/pk0tYZk7s8T+J7VLy0Da6J1+wCT0AtHg== - dependencies: - boolbase "~1.0.0" - -object-keys@^1.0.12: - version "1.1.1" - resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.1.1.tgz" - integrity sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA== - -once@^1.3.0, once@^1.3.1, once@^1.4.0: - version "1.4.0" - resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz" - integrity sha1-WDsap3WWHUsROsF9nFC6753Xa9E= - dependencies: - wrappy "1" - -p-cancelable@^1.0.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/p-cancelable/-/p-cancelable-1.1.0.tgz" - integrity sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw== - -package-json@^6.3.0: - version "6.5.0" - resolved "https://registry.yarnpkg.com/package-json/-/package-json-6.5.0.tgz" - integrity sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ== - dependencies: - got "^9.6.0" - registry-auth-token "^4.0.0" - registry-url "^5.0.0" - semver "^6.2.0" - -parse5@^3.0.1: - version "3.0.3" - resolved "https://registry.yarnpkg.com/parse5/-/parse5-3.0.3.tgz" - integrity sha512-rgO9Zg5LLLkfJF9E6CCmXlSE4UVceloys8JrFqCcHloC3usd/kJCyPDwH2SOlzix2j3xaP9sUX3e8+kvkuleAA== - dependencies: - "@types/node" "*" - -path-exists@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-4.0.0.tgz" - integrity sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w== - -path-is-absolute@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz" - integrity sha1-F0uSaHNVNP+8es5r9TpanhtcX18= - -path-parse@^1.0.6: - version "1.0.6" - resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.6.tgz" - integrity sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw== - -pend@~1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/pend/-/pend-1.2.0.tgz" - integrity sha1-elfrVQpng/kRUzH89GY9XI4AelA= - -pify@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/pify/-/pify-3.0.0.tgz" - integrity sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY= - -prepend-http@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/prepend-http/-/prepend-http-2.0.0.tgz" - integrity sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc= - -process-nextick-args@~2.0.0: - version "2.0.1" - resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.1.tgz" - integrity sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag== - -progress@^2.0.3: - version "2.0.3" - resolved "https://registry.yarnpkg.com/progress/-/progress-2.0.3.tgz" - integrity sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA== - -proto-list@~1.2.1: - version "1.2.4" - resolved "https://registry.yarnpkg.com/proto-list/-/proto-list-1.2.4.tgz" - integrity sha1-IS1b/hMYMGpCD2QCuOJv85ZHqEk= - -pump@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/pump/-/pump-3.0.0.tgz" - integrity sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww== - dependencies: - end-of-stream "^1.1.0" - once "^1.3.1" - -punycode@^2.1.0: - version "2.1.1" - resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.1.1.tgz" - integrity sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A== - -pupa@^2.0.1: - version "2.1.1" - resolved "https://registry.yarnpkg.com/pupa/-/pupa-2.1.1.tgz" - integrity sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A== - dependencies: - escape-goat "^2.0.0" - -rc@^1.2.8: - version "1.2.8" - resolved "https://registry.yarnpkg.com/rc/-/rc-1.2.8.tgz" - integrity sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw== - dependencies: - deep-extend "^0.6.0" - ini "~1.3.0" - minimist "^1.2.0" - strip-json-comments "~2.0.1" - -read-config-file@6.0.0: - version "6.0.0" - resolved "https://registry.yarnpkg.com/read-config-file/-/read-config-file-6.0.0.tgz" - integrity sha512-PHjROSdpceKUmqS06wqwP92VrM46PZSTubmNIMJ5DrMwg1OgenSTSEHIkCa6TiOJ+y/J0xnG1fFwG3M+Oi1aNA== - dependencies: - dotenv "^8.2.0" - dotenv-expand "^5.1.0" - js-yaml "^3.13.1" - json5 "^2.1.2" - lazy-val "^1.0.4" - -readable-stream@^2.2.2: - version "2.3.7" - resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.7.tgz" - integrity sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw== - dependencies: - core-util-is "~1.0.0" - inherits "~2.0.3" - isarray "~1.0.0" - process-nextick-args "~2.0.0" - safe-buffer "~5.1.1" - string_decoder "~1.1.1" - util-deprecate "~1.0.1" - -readable-stream@^3.1.1: - version "3.6.0" - resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.0.tgz" - integrity sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA== - dependencies: - inherits "^2.0.3" - string_decoder "^1.1.1" - util-deprecate "^1.0.1" - -registry-auth-token@^4.0.0: - version "4.2.1" - resolved "https://registry.yarnpkg.com/registry-auth-token/-/registry-auth-token-4.2.1.tgz" - integrity sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw== - dependencies: - rc "^1.2.8" - -registry-url@^5.0.0: - version "5.1.0" - resolved "https://registry.yarnpkg.com/registry-url/-/registry-url-5.1.0.tgz" - integrity sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw== - dependencies: - rc "^1.2.8" - -require-directory@^2.1.1: - version "2.1.1" - resolved "https://registry.yarnpkg.com/require-directory/-/require-directory-2.1.1.tgz" - integrity sha1-jGStX9MNqxyXbiNE/+f3kqam30I= - -resolve@^1.10.0: - version "1.20.0" - resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.20.0.tgz" - integrity sha512-wENBPt4ySzg4ybFQW2TT1zMQucPK95HSh/nq2CFTZVOGut2+pQvSsgtda4d26YrYcr067wjbmzOG8byDPBX63A== - dependencies: - is-core-module "^2.2.0" - path-parse "^1.0.6" - -responselike@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/responselike/-/responselike-1.0.2.tgz" - integrity sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec= - dependencies: - lowercase-keys "^1.0.0" - -rimraf@3.0.2: - version "3.0.2" - resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-3.0.2.tgz" - integrity sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA== - dependencies: - glob "^7.1.3" - -roarr@^2.15.3: - version "2.15.4" - resolved "https://registry.yarnpkg.com/roarr/-/roarr-2.15.4.tgz" - integrity sha512-CHhPh+UNHD2GTXNYhPWLnU8ONHdI+5DI+4EYIAOaiD63rHeYlZvyh8P+in5999TTSFgUYuKUAjzRI4mdh/p+2A== - dependencies: - boolean "^3.0.1" - detect-node "^2.0.4" - globalthis "^1.0.1" - json-stringify-safe "^5.0.1" - semver-compare "^1.0.0" - sprintf-js "^1.1.2" - -safe-buffer@~5.1.0, safe-buffer@~5.1.1: - version "5.1.2" - resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz" - integrity sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g== - -safe-buffer@~5.2.0: - version "5.2.1" - resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz" - integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== - -"safer-buffer@>= 2.1.2 < 3.0.0": - version "2.1.2" - resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz" - integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg== - -sanitize-filename@^1.6.3: - version "1.6.3" - resolved "https://registry.yarnpkg.com/sanitize-filename/-/sanitize-filename-1.6.3.tgz" - integrity sha512-y/52Mcy7aw3gRm7IrcGDFx/bCk4AhRh2eI9luHOQM86nZsqwiRkkq2GekHXBBD+SmPidc8i2PqtYZl+pWJ8Oeg== - dependencies: - truncate-utf8-bytes "^1.0.0" - -sax@^1.2.4: - version "1.2.4" - resolved "https://registry.yarnpkg.com/sax/-/sax-1.2.4.tgz" - integrity sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw== - -semver-compare@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/semver-compare/-/semver-compare-1.0.0.tgz" - integrity sha1-De4hahyUGrN+nvsXiPavxf9VN/w= - -semver-diff@^3.1.1: - version "3.1.1" - resolved "https://registry.yarnpkg.com/semver-diff/-/semver-diff-3.1.1.tgz" - integrity sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg== - dependencies: - semver "^6.3.0" - -"semver@2 || 3 || 4 || 5": - version "5.7.1" - resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.1.tgz" - integrity sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ== - -semver@^6.0.0, semver@^6.2.0, semver@^6.3.0: - version "6.3.0" - resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.0.tgz" - integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw== - -semver@^7.3.2: - version "7.3.4" - resolved "https://registry.yarnpkg.com/semver/-/semver-7.3.4.tgz" - integrity sha512-tCfb2WLjqFAtXn4KEdxIhalnRtoKFN7nAwj0B3ZXCbQloV2tq5eDbcTmT68JJD3nRJq24/XgxtQKFIpQdtvmVw== - dependencies: - lru-cache "^6.0.0" - -serialize-error@^7.0.1: - version "7.0.1" - resolved "https://registry.yarnpkg.com/serialize-error/-/serialize-error-7.0.1.tgz" - integrity sha512-8I8TjW5KMOKsZQTvoxjuSIa7foAwPWGOts+6o7sgjz41/qMD9VQHEDxi6PBvK2l0MXUmqZyNpUK+T2tQaaElvw== - dependencies: - type-fest "^0.13.1" - -signal-exit@^3.0.2: - version "3.0.3" - resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.3.tgz" - integrity sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA== - -slice-ansi@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-3.0.0.tgz" - integrity sha512-pSyv7bSTC7ig9Dcgbw9AuRNUb5k5V6oDudjZoMBSr13qpLBG7tB+zgCkARjq7xIUgdz5P1Qe8u+rSGdouOOIyQ== - dependencies: - ansi-styles "^4.0.0" - astral-regex "^2.0.0" - is-fullwidth-code-point "^3.0.0" - -sort-keys-length@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/sort-keys-length/-/sort-keys-length-1.0.1.tgz" - integrity sha1-nLb09OnkgVWmqgZx7dM2/xR5oYg= - dependencies: - sort-keys "^1.0.0" - -sort-keys@^1.0.0: - version "1.1.2" - resolved "https://registry.yarnpkg.com/sort-keys/-/sort-keys-1.1.2.tgz" - integrity sha1-RBttTTRnmPG05J6JIK37oOVD+a0= - dependencies: - is-plain-obj "^1.0.0" - -source-map-support@^0.5.19: - version "0.5.19" - resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.19.tgz" - integrity sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw== - dependencies: - buffer-from "^1.0.0" - source-map "^0.6.0" - -source-map@^0.6.0: - version "0.6.1" - resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz" - integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g== - -spdx-correct@^3.0.0: - version "3.1.1" - resolved "https://registry.yarnpkg.com/spdx-correct/-/spdx-correct-3.1.1.tgz" - integrity sha512-cOYcUWwhCuHCXi49RhFRCyJEK3iPj1Ziz9DpViV3tbZOwXD49QzIN3MpOLJNxh2qwq2lJJZaKMVw9qNi4jTC0w== - dependencies: - spdx-expression-parse "^3.0.0" - spdx-license-ids "^3.0.0" - -spdx-exceptions@^2.1.0: - version "2.3.0" - resolved "https://registry.yarnpkg.com/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz" - integrity sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A== - -spdx-expression-parse@^3.0.0: - version "3.0.1" - resolved "https://registry.yarnpkg.com/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz" - integrity sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q== - dependencies: - spdx-exceptions "^2.1.0" - spdx-license-ids "^3.0.0" - -spdx-license-ids@^3.0.0: - version "3.0.7" - resolved "https://registry.yarnpkg.com/spdx-license-ids/-/spdx-license-ids-3.0.7.tgz" - integrity sha512-U+MTEOO0AiDzxwFvoa4JVnMV6mZlJKk2sBLt90s7G0Gd0Mlknc7kxEn3nuDPNZRta7O2uy8oLcZLVT+4sqNZHQ== - -sprintf-js@^1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.1.2.tgz" - integrity sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug== - -sprintf-js@~1.0.2: - version "1.0.3" - resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.0.3.tgz" - integrity sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw= - -stat-mode@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/stat-mode/-/stat-mode-1.0.0.tgz" - integrity sha512-jH9EhtKIjuXZ2cWxmXS8ZP80XyC3iasQxMDV8jzhNJpfDb7VbQLVW4Wvsxz9QZvzV+G4YoSfBUVKDOyxLzi/sg== - -string-width@^3.0.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/string-width/-/string-width-3.1.0.tgz" - integrity sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w== - dependencies: - emoji-regex "^7.0.1" - is-fullwidth-code-point "^2.0.0" - strip-ansi "^5.1.0" - -string-width@^4.0.0, string-width@^4.1.0, string-width@^4.2.0: - version "4.2.2" - resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.2.tgz" - integrity sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA== - dependencies: - emoji-regex "^8.0.0" - is-fullwidth-code-point "^3.0.0" - strip-ansi "^6.0.0" - -string_decoder@^1.1.1: - version "1.3.0" - resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz" - integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA== - dependencies: - safe-buffer "~5.2.0" - -string_decoder@~1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.1.1.tgz" - integrity sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg== - dependencies: - safe-buffer "~5.1.0" - -strip-ansi@^5.1.0: - version "5.2.0" - resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-5.2.0.tgz" - integrity sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA== - dependencies: - ansi-regex "^4.1.0" - -strip-ansi@^6.0.0: - version "6.0.0" - resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.0.tgz" - integrity sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w== - dependencies: - ansi-regex "^5.0.0" - -strip-json-comments@~2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-2.0.1.tgz" - integrity sha1-PFMZQukIwml8DsNEhYwobHygpgo= - -sumchecker@^3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/sumchecker/-/sumchecker-3.0.1.tgz" - integrity sha512-MvjXzkz/BOfyVDkG0oFOtBxHX2u3gKbMHIF/dXblZsgD3BWOFLmHovIpZY7BykJdAjcqRCBi1WYBNdEC9yI7vg== - dependencies: - debug "^4.1.0" - -supports-color@^5.3.0: - version "5.5.0" - resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.5.0.tgz" - integrity sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow== - dependencies: - has-flag "^3.0.0" - -supports-color@^7.1.0: - version "7.2.0" - resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz" - integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw== - dependencies: - has-flag "^4.0.0" - -temp-file@^3.3.7: - version "3.3.7" - resolved "https://registry.yarnpkg.com/temp-file/-/temp-file-3.3.7.tgz" - integrity sha512-9tBJKt7GZAQt/Rg0QzVWA8Am8c1EFl+CAv04/aBVqlx5oyfQ508sFIABshQ0xbZu6mBrFLWIUXO/bbLYghW70g== - dependencies: - async-exit-hook "^2.0.1" - fs-extra "^8.1.0" - -term-size@^2.1.0: - version "2.2.1" - resolved "https://registry.yarnpkg.com/term-size/-/term-size-2.2.1.tgz" - integrity sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg== - -to-readable-stream@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/to-readable-stream/-/to-readable-stream-1.0.0.tgz" - integrity sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q== - -truncate-utf8-bytes@^1.0.0: - version "1.0.2" - resolved "https://registry.yarnpkg.com/truncate-utf8-bytes/-/truncate-utf8-bytes-1.0.2.tgz" - integrity sha1-QFkjkJWS1W94pYGENLC3hInKXys= - dependencies: - utf8-byte-length "^1.0.1" - -tunnel@^0.0.6: - version "0.0.6" - resolved "https://registry.yarnpkg.com/tunnel/-/tunnel-0.0.6.tgz" - integrity sha512-1h/Lnq9yajKY2PEbBadPXj3VxsDDu844OnaAo52UVmIzIvwwtBPIuNvkjuzBlTWpfJyUbG3ez0KSBibQkj4ojg== - -type-fest@^0.13.1: - version "0.13.1" - resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.13.1.tgz" - integrity sha512-34R7HTnG0XIJcBSn5XhDd7nNFPRcXYRZrBB2O2jdKqYODldSzBAqzsWoZYYvduky73toYS/ESqxPvkDf/F0XMg== - -type-fest@^0.8.1: - version "0.8.1" - resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.8.1.tgz" - integrity sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA== - -typedarray-to-buffer@^3.1.5: - version "3.1.5" - resolved "https://registry.yarnpkg.com/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz" - integrity sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q== - dependencies: - is-typedarray "^1.0.0" - -typedarray@^0.0.6: - version "0.0.6" - resolved "https://registry.yarnpkg.com/typedarray/-/typedarray-0.0.6.tgz" - integrity sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c= - -unique-string@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/unique-string/-/unique-string-2.0.0.tgz" - integrity sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg== - dependencies: - crypto-random-string "^2.0.0" - -universalify@^0.1.0: - version "0.1.2" - resolved "https://registry.yarnpkg.com/universalify/-/universalify-0.1.2.tgz" - integrity sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg== - -universalify@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/universalify/-/universalify-2.0.0.tgz" - integrity sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ== - -unused-filename@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/unused-filename/-/unused-filename-2.1.0.tgz" - integrity sha512-BMiNwJbuWmqCpAM1FqxCTD7lXF97AvfQC8Kr/DIeA6VtvhJaMDupZ82+inbjl5yVP44PcxOuCSxye1QMS0wZyg== - dependencies: - modify-filename "^1.1.0" - path-exists "^4.0.0" - -update-notifier@^4.1.1: - version "4.1.3" - resolved "https://registry.yarnpkg.com/update-notifier/-/update-notifier-4.1.3.tgz" - integrity sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A== - dependencies: - boxen "^4.2.0" - chalk "^3.0.0" - configstore "^5.0.1" - has-yarn "^2.1.0" - import-lazy "^2.1.0" - is-ci "^2.0.0" - is-installed-globally "^0.3.1" - is-npm "^4.0.0" - is-yarn-global "^0.3.0" - latest-version "^5.0.0" - pupa "^2.0.1" - semver-diff "^3.1.1" - xdg-basedir "^4.0.0" - -uri-js@^4.2.2: - version "4.4.1" - resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz" - integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg== - dependencies: - punycode "^2.1.0" - -url-parse-lax@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/url-parse-lax/-/url-parse-lax-3.0.0.tgz" - integrity sha1-FrXK/Afb42dsGxmZF3gj1lA6yww= - dependencies: - prepend-http "^2.0.0" - -utf8-byte-length@^1.0.1: - version "1.0.4" - resolved "https://registry.yarnpkg.com/utf8-byte-length/-/utf8-byte-length-1.0.4.tgz" - integrity sha1-9F8VDExm7uloGGUFq5P8u4rWv2E= - -util-deprecate@^1.0.1, util-deprecate@~1.0.1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz" - integrity sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8= - -v8-compile-cache@2.1.1: - version "2.1.1" - resolved "https://registry.yarnpkg.com/v8-compile-cache/-/v8-compile-cache-2.1.1.tgz" - integrity sha512-8OQ9CL+VWyt3JStj7HX7/ciTL2V3Rl1Wf5OL+SNTm0yK1KvtReVulksyeRnCANHHuUxHlQig+JJDlUhBt1NQDQ== - -validate-npm-package-license@^3.0.1: - version "3.0.4" - resolved "https://registry.yarnpkg.com/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz" - integrity sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew== - dependencies: - spdx-correct "^3.0.0" - spdx-expression-parse "^3.0.0" - -widest-line@^3.1.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/widest-line/-/widest-line-3.1.0.tgz" - integrity sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg== - dependencies: - string-width "^4.0.0" - -wrap-ansi@^7.0.0: - version "7.0.0" - resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz" - integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== - dependencies: - ansi-styles "^4.0.0" - string-width "^4.1.0" - strip-ansi "^6.0.0" - -wrappy@1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz" - integrity sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8= - -write-file-atomic@^3.0.0: - version "3.0.3" - resolved "https://registry.yarnpkg.com/write-file-atomic/-/write-file-atomic-3.0.3.tgz" - integrity sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q== - dependencies: - imurmurhash "^0.1.4" - is-typedarray "^1.0.0" - signal-exit "^3.0.2" - typedarray-to-buffer "^3.1.5" - -xdg-basedir@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/xdg-basedir/-/xdg-basedir-4.0.0.tgz" - integrity sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q== - -y18n@^5.0.5: - version "5.0.5" - resolved "https://registry.yarnpkg.com/y18n/-/y18n-5.0.5.tgz" - integrity sha512-hsRUr4FFrvhhRH12wOdfs38Gy7k2FFzB9qgN9v3aLykRq0dRcdcpz5C9FxdS2NuhOrI/628b/KSTJ3rwHysYSg== - -yallist@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz" - integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A== - -yargs-parser@^20.2.2: - version "20.2.7" - resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.7.tgz" - integrity sha512-FiNkvbeHzB/syOjIUxFDCnhSfzAL8R5vs40MgLFBorXACCOAEaWu0gRZl14vG8MR9AOJIZbmkjhusqBYZ3HTHw== - -yargs@^16.0.3: - version "16.2.0" - resolved "https://registry.yarnpkg.com/yargs/-/yargs-16.2.0.tgz" - integrity sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw== - dependencies: - cliui "^7.0.2" - escalade "^3.1.1" - get-caller-file "^2.0.5" - require-directory "^2.1.1" - string-width "^4.2.0" - y18n "^5.0.5" - yargs-parser "^20.2.2" - -yauzl@^2.10.0: - version "2.10.0" - resolved "https://registry.yarnpkg.com/yauzl/-/yauzl-2.10.0.tgz" - integrity sha1-x+sXyT4RLLEIb6bY5R+wZnt5pfk= - dependencies: - buffer-crc32 "~0.2.3" - fd-slicer "~1.1.0" diff --git a/vripper-gui/.gitignore b/vripper-gui/.gitignore new file mode 100644 index 00000000..549e00a2 --- /dev/null +++ b/vripper-gui/.gitignore @@ -0,0 +1,33 @@ +HELP.md +target/ +!.mvn/wrapper/maven-wrapper.jar +!**/src/main/**/target/ +!**/src/test/**/target/ + +### STS ### +.apt_generated +.classpath +.factorypath +.project +.settings +.springBeans +.sts4-cache + +### IntelliJ IDEA ### +.idea +*.iws +*.iml +*.ipr + +### NetBeans ### +/nbproject/private/ +/nbbuild/ +/dist/ +/nbdist/ +/.nb-gradle/ +build/ +!**/src/main/**/build/ +!**/src/test/**/build/ + +### VS Code ### +.vscode/ diff --git a/vripper-gui/.mvn/wrapper/maven-wrapper.jar b/vripper-gui/.mvn/wrapper/maven-wrapper.jar new file mode 100644 index 00000000..c1dd12f1 Binary files /dev/null and b/vripper-gui/.mvn/wrapper/maven-wrapper.jar differ diff --git a/vripper-gui/.mvn/wrapper/maven-wrapper.properties b/vripper-gui/.mvn/wrapper/maven-wrapper.properties new file mode 100644 index 00000000..b74bf7fc --- /dev/null +++ b/vripper-gui/.mvn/wrapper/maven-wrapper.properties @@ -0,0 +1,2 @@ +distributionUrl=https://repo.maven.apache.org/maven2/org/apache/maven/apache-maven/3.8.6/apache-maven-3.8.6-bin.zip +wrapperUrl=https://repo.maven.apache.org/maven2/org/apache/maven/wrapper/maven-wrapper/3.1.0/maven-wrapper-3.1.0.jar diff --git a/vripper-gui/mvnw b/vripper-gui/mvnw new file mode 100644 index 00000000..8a8fb228 --- /dev/null +++ b/vripper-gui/mvnw @@ -0,0 +1,316 @@ +#!/bin/sh +# ---------------------------------------------------------------------------- +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you 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 +# +# https://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. +# ---------------------------------------------------------------------------- + +# ---------------------------------------------------------------------------- +# Maven Start Up Batch script +# +# Required ENV vars: +# ------------------ +# JAVA_HOME - location of a JDK home dir +# +# Optional ENV vars +# ----------------- +# M2_HOME - location of maven2's installed home dir +# MAVEN_OPTS - parameters passed to the Java VM when running Maven +# e.g. to debug Maven itself, use +# set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000 +# MAVEN_SKIP_RC - flag to disable loading of mavenrc files +# ---------------------------------------------------------------------------- + +if [ -z "$MAVEN_SKIP_RC" ] ; then + + if [ -f /usr/local/etc/mavenrc ] ; then + . /usr/local/etc/mavenrc + fi + + if [ -f /etc/mavenrc ] ; then + . /etc/mavenrc + fi + + if [ -f "$HOME/.mavenrc" ] ; then + . "$HOME/.mavenrc" + fi + +fi + +# OS specific support. $var _must_ be set to either true or false. +cygwin=false; +darwin=false; +mingw=false +case "`uname`" in + CYGWIN*) cygwin=true ;; + MINGW*) mingw=true;; + Darwin*) darwin=true + # Use /usr/libexec/java_home if available, otherwise fall back to /Library/Java/Home + # See https://developer.apple.com/library/mac/qa/qa1170/_index.html + if [ -z "$JAVA_HOME" ]; then + if [ -x "/usr/libexec/java_home" ]; then + export JAVA_HOME="`/usr/libexec/java_home`" + else + export JAVA_HOME="/Library/Java/Home" + fi + fi + ;; +esac + +if [ -z "$JAVA_HOME" ] ; then + if [ -r /etc/gentoo-release ] ; then + JAVA_HOME=`java-config --jre-home` + fi +fi + +if [ -z "$M2_HOME" ] ; then + ## resolve links - $0 may be a link to maven's home + PRG="$0" + + # need this for relative symlinks + while [ -h "$PRG" ] ; do + ls=`ls -ld "$PRG"` + link=`expr "$ls" : '.*-> \(.*\)$'` + if expr "$link" : '/.*' > /dev/null; then + PRG="$link" + else + PRG="`dirname "$PRG"`/$link" + fi + done + + saveddir=`pwd` + + M2_HOME=`dirname "$PRG"`/.. + + # make it fully qualified + M2_HOME=`cd "$M2_HOME" && pwd` + + cd "$saveddir" + # echo Using m2 at $M2_HOME +fi + +# For Cygwin, ensure paths are in UNIX format before anything is touched +if $cygwin ; then + [ -n "$M2_HOME" ] && + M2_HOME=`cygpath --unix "$M2_HOME"` + [ -n "$JAVA_HOME" ] && + JAVA_HOME=`cygpath --unix "$JAVA_HOME"` + [ -n "$CLASSPATH" ] && + CLASSPATH=`cygpath --path --unix "$CLASSPATH"` +fi + +# For Mingw, ensure paths are in UNIX format before anything is touched +if $mingw ; then + [ -n "$M2_HOME" ] && + M2_HOME="`(cd "$M2_HOME"; pwd)`" + [ -n "$JAVA_HOME" ] && + JAVA_HOME="`(cd "$JAVA_HOME"; pwd)`" +fi + +if [ -z "$JAVA_HOME" ]; then + javaExecutable="`which javac`" + if [ -n "$javaExecutable" ] && ! [ "`expr \"$javaExecutable\" : '\([^ ]*\)'`" = "no" ]; then + # readlink(1) is not available as standard on Solaris 10. + readLink=`which readlink` + if [ ! `expr "$readLink" : '\([^ ]*\)'` = "no" ]; then + if $darwin ; then + javaHome="`dirname \"$javaExecutable\"`" + javaExecutable="`cd \"$javaHome\" && pwd -P`/javac" + else + javaExecutable="`readlink -f \"$javaExecutable\"`" + fi + javaHome="`dirname \"$javaExecutable\"`" + javaHome=`expr "$javaHome" : '\(.*\)/bin'` + JAVA_HOME="$javaHome" + export JAVA_HOME + fi + fi +fi + +if [ -z "$JAVACMD" ] ; then + if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + else + JAVACMD="`\\unset -f command; \\command -v java`" + fi +fi + +if [ ! -x "$JAVACMD" ] ; then + echo "Error: JAVA_HOME is not defined correctly." >&2 + echo " We cannot execute $JAVACMD" >&2 + exit 1 +fi + +if [ -z "$JAVA_HOME" ] ; then + echo "Warning: JAVA_HOME environment variable is not set." +fi + +CLASSWORLDS_LAUNCHER=org.codehaus.plexus.classworlds.launcher.Launcher + +# traverses directory structure from process work directory to filesystem root +# first directory with .mvn subdirectory is considered project base directory +find_maven_basedir() { + + if [ -z "$1" ] + then + echo "Path not specified to find_maven_basedir" + return 1 + fi + + basedir="$1" + wdir="$1" + while [ "$wdir" != '/' ] ; do + if [ -d "$wdir"/.mvn ] ; then + basedir=$wdir + break + fi + # workaround for JBEAP-8937 (on Solaris 10/Sparc) + if [ -d "${wdir}" ]; then + wdir=`cd "$wdir/.."; pwd` + fi + # end of workaround + done + echo "${basedir}" +} + +# concatenates all lines of a file +concat_lines() { + if [ -f "$1" ]; then + echo "$(tr -s '\n' ' ' < "$1")" + fi +} + +BASE_DIR=`find_maven_basedir "$(pwd)"` +if [ -z "$BASE_DIR" ]; then + exit 1; +fi + +########################################################################################## +# Extension to allow automatically downloading the maven-wrapper.jar from Maven-central +# This allows using the maven wrapper in projects that prohibit checking in binary data. +########################################################################################## +if [ -r "$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" ]; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found .mvn/wrapper/maven-wrapper.jar" + fi +else + if [ "$MVNW_VERBOSE" = true ]; then + echo "Couldn't find .mvn/wrapper/maven-wrapper.jar, downloading it ..." + fi + if [ -n "$MVNW_REPOURL" ]; then + jarUrl="$MVNW_REPOURL/org/apache/maven/wrapper/maven-wrapper/3.1.0/maven-wrapper-3.1.0.jar" + else + jarUrl="https://repo.maven.apache.org/maven2/org/apache/maven/wrapper/maven-wrapper/3.1.0/maven-wrapper-3.1.0.jar" + fi + while IFS="=" read key value; do + case "$key" in (wrapperUrl) jarUrl="$value"; break ;; + esac + done < "$BASE_DIR/.mvn/wrapper/maven-wrapper.properties" + if [ "$MVNW_VERBOSE" = true ]; then + echo "Downloading from: $jarUrl" + fi + wrapperJarPath="$BASE_DIR/.mvn/wrapper/maven-wrapper.jar" + if $cygwin; then + wrapperJarPath=`cygpath --path --windows "$wrapperJarPath"` + fi + + if command -v wget > /dev/null; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found wget ... using wget" + fi + if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then + wget "$jarUrl" -O "$wrapperJarPath" || rm -f "$wrapperJarPath" + else + wget --http-user=$MVNW_USERNAME --http-password=$MVNW_PASSWORD "$jarUrl" -O "$wrapperJarPath" || rm -f "$wrapperJarPath" + fi + elif command -v curl > /dev/null; then + if [ "$MVNW_VERBOSE" = true ]; then + echo "Found curl ... using curl" + fi + if [ -z "$MVNW_USERNAME" ] || [ -z "$MVNW_PASSWORD" ]; then + curl -o "$wrapperJarPath" "$jarUrl" -f + else + curl --user $MVNW_USERNAME:$MVNW_PASSWORD -o "$wrapperJarPath" "$jarUrl" -f + fi + + else + if [ "$MVNW_VERBOSE" = true ]; then + echo "Falling back to using Java to download" + fi + javaClass="$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.java" + # For Cygwin, switch paths to Windows format before running javac + if $cygwin; then + javaClass=`cygpath --path --windows "$javaClass"` + fi + if [ -e "$javaClass" ]; then + if [ ! -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then + if [ "$MVNW_VERBOSE" = true ]; then + echo " - Compiling MavenWrapperDownloader.java ..." + fi + # Compiling the Java class + ("$JAVA_HOME/bin/javac" "$javaClass") + fi + if [ -e "$BASE_DIR/.mvn/wrapper/MavenWrapperDownloader.class" ]; then + # Running the downloader + if [ "$MVNW_VERBOSE" = true ]; then + echo " - Running MavenWrapperDownloader.java ..." + fi + ("$JAVA_HOME/bin/java" -cp .mvn/wrapper MavenWrapperDownloader "$MAVEN_PROJECTBASEDIR") + fi + fi + fi +fi +########################################################################################## +# End of extension +########################################################################################## + +export MAVEN_PROJECTBASEDIR=${MAVEN_BASEDIR:-"$BASE_DIR"} +if [ "$MVNW_VERBOSE" = true ]; then + echo $MAVEN_PROJECTBASEDIR +fi +MAVEN_OPTS="$(concat_lines "$MAVEN_PROJECTBASEDIR/.mvn/jvm.config") $MAVEN_OPTS" + +# For Cygwin, switch paths to Windows format before running java +if $cygwin; then + [ -n "$M2_HOME" ] && + M2_HOME=`cygpath --path --windows "$M2_HOME"` + [ -n "$JAVA_HOME" ] && + JAVA_HOME=`cygpath --path --windows "$JAVA_HOME"` + [ -n "$CLASSPATH" ] && + CLASSPATH=`cygpath --path --windows "$CLASSPATH"` + [ -n "$MAVEN_PROJECTBASEDIR" ] && + MAVEN_PROJECTBASEDIR=`cygpath --path --windows "$MAVEN_PROJECTBASEDIR"` +fi + +# Provide a "standardized" way to retrieve the CLI args that will +# work with both Windows and non-Windows executions. +MAVEN_CMD_LINE_ARGS="$MAVEN_CONFIG $@" +export MAVEN_CMD_LINE_ARGS + +WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain + +exec "$JAVACMD" \ + $MAVEN_OPTS \ + $MAVEN_DEBUG_OPTS \ + -classpath "$MAVEN_PROJECTBASEDIR/.mvn/wrapper/maven-wrapper.jar" \ + "-Dmaven.home=${M2_HOME}" \ + "-Dmaven.multiModuleProjectDirectory=${MAVEN_PROJECTBASEDIR}" \ + ${WRAPPER_LAUNCHER} $MAVEN_CONFIG "$@" diff --git a/vripper-gui/mvnw.cmd b/vripper-gui/mvnw.cmd new file mode 100644 index 00000000..1d8ab018 --- /dev/null +++ b/vripper-gui/mvnw.cmd @@ -0,0 +1,188 @@ +@REM ---------------------------------------------------------------------------- +@REM Licensed to the Apache Software Foundation (ASF) under one +@REM or more contributor license agreements. See the NOTICE file +@REM distributed with this work for additional information +@REM regarding copyright ownership. The ASF licenses this file +@REM to you under the Apache License, Version 2.0 (the +@REM "License"); you may not use this file except in compliance +@REM with the License. You may obtain a copy of the License at +@REM +@REM https://www.apache.org/licenses/LICENSE-2.0 +@REM +@REM Unless required by applicable law or agreed to in writing, +@REM software distributed under the License is distributed on an +@REM "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +@REM KIND, either express or implied. See the License for the +@REM specific language governing permissions and limitations +@REM under the License. +@REM ---------------------------------------------------------------------------- + +@REM ---------------------------------------------------------------------------- +@REM Maven Start Up Batch script +@REM +@REM Required ENV vars: +@REM JAVA_HOME - location of a JDK home dir +@REM +@REM Optional ENV vars +@REM M2_HOME - location of maven2's installed home dir +@REM MAVEN_BATCH_ECHO - set to 'on' to enable the echoing of the batch commands +@REM MAVEN_BATCH_PAUSE - set to 'on' to wait for a keystroke before ending +@REM MAVEN_OPTS - parameters passed to the Java VM when running Maven +@REM e.g. to debug Maven itself, use +@REM set MAVEN_OPTS=-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=8000 +@REM MAVEN_SKIP_RC - flag to disable loading of mavenrc files +@REM ---------------------------------------------------------------------------- + +@REM Begin all REM lines with '@' in case MAVEN_BATCH_ECHO is 'on' +@echo off +@REM set title of command window +title %0 +@REM enable echoing by setting MAVEN_BATCH_ECHO to 'on' +@if "%MAVEN_BATCH_ECHO%" == "on" echo %MAVEN_BATCH_ECHO% + +@REM set %HOME% to equivalent of $HOME +if "%HOME%" == "" (set "HOME=%HOMEDRIVE%%HOMEPATH%") + +@REM Execute a user defined script before this one +if not "%MAVEN_SKIP_RC%" == "" goto skipRcPre +@REM check for pre script, once with legacy .bat ending and once with .cmd ending +if exist "%USERPROFILE%\mavenrc_pre.bat" call "%USERPROFILE%\mavenrc_pre.bat" %* +if exist "%USERPROFILE%\mavenrc_pre.cmd" call "%USERPROFILE%\mavenrc_pre.cmd" %* +:skipRcPre + +@setlocal + +set ERROR_CODE=0 + +@REM To isolate internal variables from possible post scripts, we use another setlocal +@setlocal + +@REM ==== START VALIDATION ==== +if not "%JAVA_HOME%" == "" goto OkJHome + +echo. +echo Error: JAVA_HOME not found in your environment. >&2 +echo Please set the JAVA_HOME variable in your environment to match the >&2 +echo location of your Java installation. >&2 +echo. +goto error + +:OkJHome +if exist "%JAVA_HOME%\bin\java.exe" goto init + +echo. +echo Error: JAVA_HOME is set to an invalid directory. >&2 +echo JAVA_HOME = "%JAVA_HOME%" >&2 +echo Please set the JAVA_HOME variable in your environment to match the >&2 +echo location of your Java installation. >&2 +echo. +goto error + +@REM ==== END VALIDATION ==== + +:init + +@REM Find the project base dir, i.e. the directory that contains the folder ".mvn". +@REM Fallback to current working directory if not found. + +set MAVEN_PROJECTBASEDIR=%MAVEN_BASEDIR% +IF NOT "%MAVEN_PROJECTBASEDIR%"=="" goto endDetectBaseDir + +set EXEC_DIR=%CD% +set WDIR=%EXEC_DIR% +:findBaseDir +IF EXIST "%WDIR%"\.mvn goto baseDirFound +cd .. +IF "%WDIR%"=="%CD%" goto baseDirNotFound +set WDIR=%CD% +goto findBaseDir + +:baseDirFound +set MAVEN_PROJECTBASEDIR=%WDIR% +cd "%EXEC_DIR%" +goto endDetectBaseDir + +:baseDirNotFound +set MAVEN_PROJECTBASEDIR=%EXEC_DIR% +cd "%EXEC_DIR%" + +:endDetectBaseDir + +IF NOT EXIST "%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config" goto endReadAdditionalConfig + +@setlocal EnableExtensions EnableDelayedExpansion +for /F "usebackq delims=" %%a in ("%MAVEN_PROJECTBASEDIR%\.mvn\jvm.config") do set JVM_CONFIG_MAVEN_PROPS=!JVM_CONFIG_MAVEN_PROPS! %%a +@endlocal & set JVM_CONFIG_MAVEN_PROPS=%JVM_CONFIG_MAVEN_PROPS% + +:endReadAdditionalConfig + +SET MAVEN_JAVA_EXE="%JAVA_HOME%\bin\java.exe" +set WRAPPER_JAR="%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.jar" +set WRAPPER_LAUNCHER=org.apache.maven.wrapper.MavenWrapperMain + +set DOWNLOAD_URL="https://repo.maven.apache.org/maven2/org/apache/maven/wrapper/maven-wrapper/3.1.0/maven-wrapper-3.1.0.jar" + +FOR /F "usebackq tokens=1,2 delims==" %%A IN ("%MAVEN_PROJECTBASEDIR%\.mvn\wrapper\maven-wrapper.properties") DO ( + IF "%%A"=="wrapperUrl" SET DOWNLOAD_URL=%%B +) + +@REM Extension to allow automatically downloading the maven-wrapper.jar from Maven-central +@REM This allows using the maven wrapper in projects that prohibit checking in binary data. +if exist %WRAPPER_JAR% ( + if "%MVNW_VERBOSE%" == "true" ( + echo Found %WRAPPER_JAR% + ) +) else ( + if not "%MVNW_REPOURL%" == "" ( + SET DOWNLOAD_URL="%MVNW_REPOURL%/org/apache/maven/wrapper/maven-wrapper/3.1.0/maven-wrapper-3.1.0.jar" + ) + if "%MVNW_VERBOSE%" == "true" ( + echo Couldn't find %WRAPPER_JAR%, downloading it ... + echo Downloading from: %DOWNLOAD_URL% + ) + + powershell -Command "&{"^ + "$webclient = new-object System.Net.WebClient;"^ + "if (-not ([string]::IsNullOrEmpty('%MVNW_USERNAME%') -and [string]::IsNullOrEmpty('%MVNW_PASSWORD%'))) {"^ + "$webclient.Credentials = new-object System.Net.NetworkCredential('%MVNW_USERNAME%', '%MVNW_PASSWORD%');"^ + "}"^ + "[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; $webclient.DownloadFile('%DOWNLOAD_URL%', '%WRAPPER_JAR%')"^ + "}" + if "%MVNW_VERBOSE%" == "true" ( + echo Finished downloading %WRAPPER_JAR% + ) +) +@REM End of extension + +@REM Provide a "standardized" way to retrieve the CLI args that will +@REM work with both Windows and non-Windows executions. +set MAVEN_CMD_LINE_ARGS=%* + +%MAVEN_JAVA_EXE% ^ + %JVM_CONFIG_MAVEN_PROPS% ^ + %MAVEN_OPTS% ^ + %MAVEN_DEBUG_OPTS% ^ + -classpath %WRAPPER_JAR% ^ + "-Dmaven.multiModuleProjectDirectory=%MAVEN_PROJECTBASEDIR%" ^ + %WRAPPER_LAUNCHER% %MAVEN_CONFIG% %* +if ERRORLEVEL 1 goto error +goto end + +:error +set ERROR_CODE=1 + +:end +@endlocal & set ERROR_CODE=%ERROR_CODE% + +if not "%MAVEN_SKIP_RC%"=="" goto skipRcPost +@REM check for post script, once with legacy .bat ending and once with .cmd ending +if exist "%USERPROFILE%\mavenrc_post.bat" call "%USERPROFILE%\mavenrc_post.bat" +if exist "%USERPROFILE%\mavenrc_post.cmd" call "%USERPROFILE%\mavenrc_post.cmd" +:skipRcPost + +@REM pause the script if MAVEN_BATCH_PAUSE is set to 'on' +if "%MAVEN_BATCH_PAUSE%"=="on" pause + +if "%MAVEN_TERMINATE_CMD%"=="on" exit %ERROR_CODE% + +cmd /C exit /B %ERROR_CODE% diff --git a/vripper-gui/pom.xml b/vripper-gui/pom.xml new file mode 100644 index 00000000..1f518dec --- /dev/null +++ b/vripper-gui/pom.xml @@ -0,0 +1,154 @@ + + + 4.0.0 + + spring-boot-starter-parent + org.springframework.boot + + 3.0.6 + + me.mnlr.vripper + vripper-gui + 4.0.0 + vripper-gui + vripper-gui + + + spring-boot-starter + org.springframework.boot + + + kotlin-reflect + org.jetbrains.kotlin + + + kotlin-stdlib + org.jetbrains.kotlin + + + javafx-base + org.openjfx + ${javafx.version} + + + javafx-base + win + org.openjfx + ${javafx.version} + + + javafx-base + linux + org.openjfx + ${javafx.version} + + + javafx-base + mac + org.openjfx + ${javafx.version} + + + javafx-controls + org.openjfx + ${javafx.version} + + + javafx-controls + win + org.openjfx + ${javafx.version} + + + javafx-controls + linux + org.openjfx + ${javafx.version} + + + javafx-controls + mac + org.openjfx + ${javafx.version} + + + javafx-graphics + win + org.openjfx + ${javafx.version} + + + javafx-graphics + linux + org.openjfx + ${javafx.version} + + + javafx-graphics + mac + org.openjfx + ${javafx.version} + + + tornadofx + no.tornado + 1.7.20 + + + vripper-core + me.mnlr.vripper + ${project.version} + + + + spring-boot-starter-test + org.springframework.boot + test + + + + 17 + 1.8.21 + 20.0.1 + false + + + ${project.basedir}/src/main/kotlin + ${project.basedir}/src/test/kotlin + + + org.springframework.boot + spring-boot-maven-plugin + + + org.jetbrains.kotlin + kotlin-maven-plugin + + + -Xjsr305=strict + + + spring + + + + + kotlin-maven-allopen + org.jetbrains.kotlin + ${kotlin.version} + + + + + + + + + github + GitHub death-claw Apache Maven Packages + https://maven.pkg.github.com/death-claw/vripper-project + + + diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/VripperGuiApplication.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/VripperGuiApplication.kt new file mode 100644 index 00000000..7271c9cf --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/VripperGuiApplication.kt @@ -0,0 +1,76 @@ +package me.mnlr.vripper + +import javafx.application.Application +import javafx.scene.image.Image +import javafx.stage.Stage +import javafx.stage.WindowEvent +import org.springframework.boot.SpringApplication +import org.springframework.boot.autoconfigure.SpringBootApplication +import org.springframework.context.ConfigurableApplicationContext +import me.mnlr.vripper.event.ApplicationInitialized +import me.mnlr.vripper.gui.Styles +import me.mnlr.vripper.view.LoadingView +import tornadofx.* +import kotlin.reflect.KClass +import kotlin.system.exitProcess + +@SpringBootApplication +class VripperGuiApplication : App( + LoadingView::class, Styles::class +) { //The application class must be a TornadoFX application and it must have the main view + + private lateinit var context: ConfigurableApplicationContext //We are going to set application context here + private var initialized = false + override fun start(stage: Stage) { + with(stage) { + width = 1200.0 + height = 600.0 + icons.addAll( + listOf( + Image("icons/16x16.png"), + Image("icons/32x32.png"), + Image("icons/48x48.png"), + Image("icons/64x64.png"), + Image("icons/128x128.png"), + Image("icons/256x256.png"), + Image("icons/512x512.png"), + Image("icons/1024x1024.png") + ) + ) + } + stage.addEventFilter(WindowEvent.WINDOW_SHOWN) { + val contextInitThread = Thread { + context = + SpringApplication.run(this.javaClass) //We start the application context and let Spring Boot to initilaize itself + context.autowireCapableBeanFactory.autowireBean(this) //We ask the context to inject all needed dependencies into the current instence (if needed) + + FX.dicontainer = object : + DIContainer { // Here we have to implement an interface for TornadoFX DI support + override fun getInstance(type: KClass): T = + context.getBean(type.java) // We find dependencies directly in Spring's application context + + override fun getInstance(type: KClass, name: String): T = + context.getBean(name, type.java) + } + fire(ApplicationInitialized) + initialized = true + } + contextInitThread.apply { + this.name = "Spring init Thread" + }.start() + } + super.start(stage) + } + + override fun stop() { // On stop, we have to stop spring as well + super.stop() + if (initialized) { + context.close() + } + exitProcess(0) + } +} + +fun main(args: Array) { + Application.launch(me.mnlr.vripper.VripperGuiApplication::class.java, *args) +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/GlobalStateController.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/GlobalStateController.kt new file mode 100644 index 00000000..1c9c3bd4 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/GlobalStateController.kt @@ -0,0 +1,62 @@ +package me.mnlr.vripper.controller + +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.formatSI +import me.mnlr.vripper.model.DownloadSpeed +import me.mnlr.vripper.model.GlobalState +import me.mnlr.vripper.model.GlobalStateModel +import me.mnlr.vripper.services.GlobalStateService +import tornadofx.* + +class GlobalStateController : Controller() { + + private val globalStateService: GlobalStateService by di() + private val eventBus: EventBus by di() + + var globalState: GlobalStateModel = GlobalStateModel(0, 0, 0, "", "") + + init { + val currentState = globalStateService.get() + globalState.apply { + running = currentState.running + remaining = currentState.remaining + error = currentState.error + loggedUser = currentState.loggedUser + downloadSpeed = currentState.downloadSpeed + } + + eventBus.flux().subscribe { + if (it!!.kind == Event.Kind.DOWNLOAD_STATUS + ) { + val newState = it.data as GlobalState + globalState.apply { + running = newState.running + remaining = newState.remaining + error = newState.error + } + } + } + + eventBus.flux().subscribe { + if (it!!.kind == Event.Kind.VG_USER + ) { + val user = it.data as String + globalState.apply { + loggedUser = user + } + } + } + + eventBus.flux().subscribe { + if (it!!.kind == Event.Kind.BYTES_PER_SECOND + ) { + val speed = + DownloadSpeed((it.data as Long).formatSI()) + globalState.apply { + downloadSpeed = speed.speed + } + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/ImageController.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/ImageController.kt new file mode 100644 index 00000000..e06dc065 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/ImageController.kt @@ -0,0 +1,31 @@ +package me.mnlr.vripper.controller + +import me.mnlr.vripper.entities.ImageDownloadState +import me.mnlr.vripper.model.ImageModel +import me.mnlr.vripper.repositories.ImageRepository +import tornadofx.Controller +import java.util.* + +class ImageController : Controller() { + + private val imageRepository: ImageRepository by di() + + fun findImages(postId: String): List { + return imageRepository.findByPostId(postId).map(::mapper) + } + + fun findImageById(id: Long): Optional { + return imageRepository.findById(id).map(::mapper) + } + + private fun mapper(it: ImageDownloadState): ImageModel { + return ImageModel( + it.id!!, + it.index + 1, + it.url, + if (it.current == 0L && it.total == 0L) 0.0 else (it.current.toDouble() / it.total), + it.status.stringValue, + it.postId + ) + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/LogController.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/LogController.kt new file mode 100644 index 00000000..39343130 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/LogController.kt @@ -0,0 +1,37 @@ +package me.mnlr.vripper.controller + +import me.mnlr.vripper.AppEndpointService +import me.mnlr.vripper.entities.LogEvent +import me.mnlr.vripper.model.LogModel +import me.mnlr.vripper.repositories.LogEventRepository +import tornadofx.Controller +import java.time.format.DateTimeFormatter +import java.util.* + +class LogController : Controller() { + private val logEventRepository: LogEventRepository by di() + private val appEndpointService: AppEndpointService by di() + private val dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss") + + fun findAll(): List { + return logEventRepository.findAll().map(::mapper) + } + + fun clear() { + appEndpointService.logClear() + } + + fun find(id: Long): Optional { + return logEventRepository.findById(id).map(::mapper) + } + + private fun mapper(it: LogEvent): LogModel { + return LogModel( + it.id!!, + it.type.stringValue, + it.status.stringValue, + it.time.format(dateTimeFormatter), + it.message + ) + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/PostController.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/PostController.kt new file mode 100644 index 00000000..44d749f0 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/PostController.kt @@ -0,0 +1,69 @@ +package me.mnlr.vripper.controller + +import me.mnlr.vripper.AppEndpointService +import me.mnlr.vripper.entities.PostDownloadState +import me.mnlr.vripper.model.PostModel +import me.mnlr.vripper.repositories.PostDownloadStateRepository +import tornadofx.Controller +import java.time.format.DateTimeFormatter +import java.util.* + +class PostController : Controller() { + + private val postDownloadStateRepository: PostDownloadStateRepository by di() + private val appEndpointService: AppEndpointService by di() + private val dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss") + + fun scan(postLinks: String) { + appEndpointService.scanLinks(postLinks) + } + + fun start(postIdList: List) { + appEndpointService.restartAll(postIdList) + } + + fun startAll() { + appEndpointService.restartAll() + } + + fun delete(postIdList: List) { + appEndpointService.remove(postIdList) + } + + fun stop(postIdList: List) { + appEndpointService.stopAll(postIdList) + + } + + fun clearPosts(): List { + return appEndpointService.clearCompleted() + } + + fun stopAll() { + appEndpointService.stopAll(null) + } + + fun findAllPosts(): List { + return postDownloadStateRepository.findAll().map(::mapper) + } + + fun findById(id: Long): Optional { + return postDownloadStateRepository.findById(id).map(::mapper) + } + + private fun mapper(it: PostDownloadState): PostModel { + return PostModel( + it.postId, + it.postTitle, + if (it.done == 0 && it.total == 0) 0.0 else (it.done.toDouble() / it.total), + it.status.stringValue, + it.url, + it.done, + it.total, + it.hosts.joinToString(separator = "||"), + it.addedOn.format(dateTimeFormatter), + it.rank + 1, + it.downloadDirectory + ) + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/SettingsController.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/SettingsController.kt new file mode 100644 index 00000000..c3fb9295 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/SettingsController.kt @@ -0,0 +1,69 @@ +package me.mnlr.vripper.controller + +import javafx.scene.control.Alert +import me.mnlr.vripper.exception.ValidationException +import me.mnlr.vripper.model.ConnectionSettings +import me.mnlr.vripper.model.DownloadSettings +import me.mnlr.vripper.model.settings.DownloadSettingsModel +import me.mnlr.vripper.model.Settings +import me.mnlr.vripper.model.ViperSettings +import me.mnlr.vripper.model.settings.ConnectionSettingsModel +import me.mnlr.vripper.model.settings.ViperSettingsModel +import me.mnlr.vripper.services.SettingsService +import tornadofx.Controller +import tornadofx.alert +import tornadofx.warning + +class SettingsController : Controller() { + + private val settingsService: SettingsService by di() + + fun findDownloadSettings(): DownloadSettings { + return settingsService.settings.downloadSettings + } + + fun findConnectionSettings(): ConnectionSettings { + return settingsService.settings.connectionSettings + } + + fun findViperGirlsSettings(): ViperSettings { + return settingsService.settings.viperSettings + } + + fun saveNewSettings( + downloadSettingsModel: DownloadSettingsModel, + connectionSettingsModel: ConnectionSettingsModel, + viperSettingsModel: ViperSettingsModel + ) { + settingsService.newSettings(Settings().apply { + downloadSettings = DownloadSettings( + downloadSettingsModel.downloadPath, + downloadSettingsModel.autoStart, + downloadSettingsModel.autoQueueThreshold, + downloadSettingsModel.forceOrder, + downloadSettingsModel.forumSubfolder, + downloadSettingsModel.threadSubLocation, + downloadSettingsModel.clearCompleted, + downloadSettingsModel.appendPostId + ) + connectionSettings = ConnectionSettings( + connectionSettingsModel.maxThreads, + connectionSettingsModel.maxTotalThreads, + connectionSettingsModel.timeout, + connectionSettingsModel.maxAttempts, + ) + viperSettings = ViperSettings( + viperSettingsModel.login, + viperSettingsModel.username, + viperSettingsModel.password, + viperSettingsModel.thanks, + viperSettingsModel.host, + ) + }) + + } + + fun getProxies(): List { + return settingsService.getProxies() + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/ThreadController.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/ThreadController.kt new file mode 100644 index 00000000..396f996a --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/controller/ThreadController.kt @@ -0,0 +1,61 @@ +package me.mnlr.vripper.controller + +import me.mnlr.vripper.AppEndpointService +import me.mnlr.vripper.entities.Thread +import me.mnlr.vripper.model.ThreadModel +import me.mnlr.vripper.model.ThreadSelectionModel +import me.mnlr.vripper.repositories.ThreadRepository +import tornadofx.* +import java.util.* + +class ThreadController : Controller() { + private val threadRepository: ThreadRepository by di() + + private val appEndpointService: AppEndpointService by di() + + fun findAll(): List { + return threadRepository.findAll().map(::threadModelMapper) + } + + fun find(id: Long): Optional { + return threadRepository.findById(id).map(::threadModelMapper) + } + + private fun threadModelMapper(it: Thread): ThreadModel { + return ThreadModel( + it.link, + it.total, + it.threadId + ) + } + + fun delete(threadIdList: List) { + appEndpointService.threadRemove(threadIdList) + } + + fun clearAll() { + appEndpointService.threadClear() + } + + fun grab(threadId: String): List { + return appEndpointService.grab(threadId).map { + ThreadSelectionModel( + it.number, + it.title, + it.url, + it.hosts, + it.postId, + it.threadId + ) + } + } + + fun download(selectedItems: List) { + appEndpointService.download(selectedItems.map { + Pair( + it.threadId, + it.postId + ) + }) + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/event/ApplicationInitialized.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/event/ApplicationInitialized.kt new file mode 100644 index 00000000..a9bd49f1 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/event/ApplicationInitialized.kt @@ -0,0 +1,6 @@ +package me.mnlr.vripper.event + +import tornadofx.* +import tornadofx.EventBus + +object ApplicationInitialized : FXEvent(EventBus.RunOn.BackgroundThread) \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/gui/Styles.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/gui/Styles.kt new file mode 100644 index 00000000..9a2e7c51 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/gui/Styles.kt @@ -0,0 +1,30 @@ +package me.mnlr.vripper.gui + +import javafx.scene.paint.Color +import javafx.scene.text.FontWeight +import tornadofx.* + +class Styles : Stylesheet() { + companion object { + val heading by cssclass() + val actionBarButton by cssclass() + } + + init { + label and heading { + padding = box(20.px) + fontSize = 20.px + fontWeight = FontWeight.BOLD + } + + actionBarButton { + prefWidth = 32.px + prefHeight = 32.px + padding = box(0.px) + backgroundColor += Color.TRANSPARENT + and(hover) { + backgroundColor += Color.LIGHTGREY + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/GlobalStateModel.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/GlobalStateModel.kt new file mode 100644 index 00000000..2fc0ea61 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/GlobalStateModel.kt @@ -0,0 +1,29 @@ +package me.mnlr.vripper.model + +import javafx.beans.property.SimpleIntegerProperty +import javafx.beans.property.SimpleStringProperty +import tornadofx.getValue +import tornadofx.setValue + +class GlobalStateModel( + running: Int, + remaining: Int, + error: Int, + loggedUser: String, + downloadSpeed: String +) { + val runningProperty = SimpleIntegerProperty(running) + var running: Int by runningProperty + + val remainingProperty = SimpleIntegerProperty(remaining) + var remaining: Int by remainingProperty + + val errorProperty = SimpleIntegerProperty(error) + var error: Int by errorProperty + + val loggedUserProperty = SimpleStringProperty(loggedUser) + var loggedUser: String by loggedUserProperty + + val downloadSpeedProperty = SimpleStringProperty(downloadSpeed) + var downloadSpeed: String by downloadSpeedProperty +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/ImageModel.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/ImageModel.kt new file mode 100644 index 00000000..c555de0d --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/ImageModel.kt @@ -0,0 +1,31 @@ +package me.mnlr.vripper.model + +import javafx.beans.property.SimpleDoubleProperty +import javafx.beans.property.SimpleIntegerProperty +import javafx.beans.property.SimpleLongProperty +import javafx.beans.property.SimpleStringProperty +import tornadofx.* + +class ImageModel( + id: Long, + index: Int, + url: String, + progress: Double, + status: String, + val postId: String +) { + val idProperty = SimpleLongProperty(id) + var id: Long by idProperty + + val indexProperty = SimpleIntegerProperty(index) + var index: Int by indexProperty + + val urlProperty = SimpleStringProperty(url) + var url: String by urlProperty + + val progressProperty = SimpleDoubleProperty(progress) + var progress: Double by progressProperty + + val statusProperty = SimpleStringProperty(status) + var status: String by statusProperty +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/LogModel.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/LogModel.kt new file mode 100644 index 00000000..92112ab1 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/LogModel.kt @@ -0,0 +1,28 @@ +package me.mnlr.vripper.model + +import javafx.beans.property.SimpleLongProperty +import javafx.beans.property.SimpleStringProperty +import tornadofx.* + +class LogModel( + id: Long, + type: String, + status: String, + time: String, + message: String +) { + val idProperty = SimpleLongProperty(id) + var id: Long by idProperty + + val typeProperty = SimpleStringProperty(type) + var type: String by typeProperty + + val statusProperty = SimpleStringProperty(status) + var status: String by statusProperty + + val timeProperty = SimpleStringProperty(time) + var time: String by timeProperty + + val messageProperty = SimpleStringProperty(message) + var message: String by messageProperty +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/PostModel.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/PostModel.kt new file mode 100644 index 00000000..feff8b00 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/PostModel.kt @@ -0,0 +1,53 @@ +package me.mnlr.vripper.model + +import javafx.beans.property.SimpleDoubleProperty +import javafx.beans.property.SimpleIntegerProperty +import javafx.beans.property.SimpleStringProperty +import tornadofx.* + +class PostModel( + postId: String, + title: String, + progress: Double, + status: String, + url: String, + done: Int, + total: Int, + hosts: String, + addedOn: String, + order: Int, + path: String +) { + val postIdProperty = SimpleStringProperty(postId) + var postId: String by postIdProperty + + val titleProperty = SimpleStringProperty(title) + var title: String by titleProperty + + val progressProperty = SimpleDoubleProperty(progress) + var progress: Double by progressProperty + + val statusProperty = SimpleStringProperty(status) + var status: String by statusProperty + + val urlProperty = SimpleStringProperty(url) + var url: String by urlProperty + + val doneProperty = SimpleIntegerProperty(done) + var done: Int by doneProperty + + val totalProperty = SimpleIntegerProperty(total) + var total: Int by totalProperty + + val hostsProperty = SimpleStringProperty(hosts) + var hosts: String by hostsProperty + + val addedOnProperty = SimpleStringProperty(addedOn) + var addedOn: String by addedOnProperty + + val orderProperty = SimpleIntegerProperty(order) + var order: Int by orderProperty + + val pathProperty = SimpleStringProperty(path) + var path: String by pathProperty +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/ThreadModel.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/ThreadModel.kt new file mode 100644 index 00000000..c7710bf7 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/ThreadModel.kt @@ -0,0 +1,17 @@ +package me.mnlr.vripper.model + +import javafx.beans.property.SimpleIntegerProperty +import javafx.beans.property.SimpleStringProperty +import tornadofx.* + +class ThreadModel( + link: String, + total: Int, + val threadId: String +) { + val linkProperty = SimpleStringProperty(link) + var link: String by linkProperty + + val totalProperty = SimpleIntegerProperty(total) + var total: Int by totalProperty +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/ThreadSelectionModel.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/ThreadSelectionModel.kt new file mode 100644 index 00000000..d3f11cee --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/ThreadSelectionModel.kt @@ -0,0 +1,27 @@ +package me.mnlr.vripper.model + +import javafx.beans.property.SimpleIntegerProperty +import javafx.beans.property.SimpleStringProperty +import me.mnlr.vripper.host.Host +import tornadofx.* + +class ThreadSelectionModel( + index: Int, + title: String, + url: String, + hosts: Map, + val postId: String, + val threadId: String +) { + val indexProperty = SimpleIntegerProperty(index) + var index: Int by indexProperty + + val titleProperty = SimpleStringProperty(title) + var title: String by titleProperty + + val urlProperty = SimpleStringProperty(url) + var url: String by urlProperty + + val hostsProperty = SimpleStringProperty(hosts.keys.map { it.host }.joinToString(", ")) + var hosts: String by hostsProperty +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/settings/ConnectionSettingsModel.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/settings/ConnectionSettingsModel.kt new file mode 100644 index 00000000..75f5c2b4 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/settings/ConnectionSettingsModel.kt @@ -0,0 +1,18 @@ +package me.mnlr.vripper.model.settings + +import javafx.beans.property.* +import tornadofx.* + +class ConnectionSettingsModel { + val maxThreadsProperty = SimpleIntegerProperty() + var maxThreads: Int by maxThreadsProperty + + val maxTotalThreadsProperty = SimpleIntegerProperty() + var maxTotalThreads: Int by maxTotalThreadsProperty + + val timeoutProperty = SimpleIntegerProperty() + var timeout: Int by timeoutProperty + + val maxAttemptsProperty = SimpleIntegerProperty() + var maxAttempts: Int by maxAttemptsProperty +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/settings/DownloadSettingsModel.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/settings/DownloadSettingsModel.kt new file mode 100644 index 00000000..8f02e901 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/settings/DownloadSettingsModel.kt @@ -0,0 +1,30 @@ +package me.mnlr.vripper.model.settings + +import javafx.beans.property.* +import tornadofx.* + +class DownloadSettingsModel { + val downloadPathProperty = SimpleStringProperty() + var downloadPath: String by downloadPathProperty + + val autoStartProperty = SimpleBooleanProperty() + var autoStart: Boolean by autoStartProperty + + val autoQueueThresholdProperty = SimpleIntegerProperty() + var autoQueueThreshold: Int by autoQueueThresholdProperty + + val forceOrderProperty = SimpleBooleanProperty() + var forceOrder: Boolean by forceOrderProperty + + val forumSubfolderProperty = SimpleBooleanProperty() + var forumSubfolder: Boolean by forumSubfolderProperty + + val threadSubLocationProperty = SimpleBooleanProperty() + var threadSubLocation: Boolean by threadSubLocationProperty + + val clearCompletedProperty = SimpleBooleanProperty() + var clearCompleted: Boolean by clearCompletedProperty + + val appendPostIdProperty = SimpleBooleanProperty() + var appendPostId: Boolean by appendPostIdProperty +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/settings/ViperSettingsModel.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/settings/ViperSettingsModel.kt new file mode 100644 index 00000000..a4f1f414 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/model/settings/ViperSettingsModel.kt @@ -0,0 +1,21 @@ +package me.mnlr.vripper.model.settings + +import javafx.beans.property.* +import tornadofx.* + +class ViperSettingsModel { + val loginProperty = SimpleBooleanProperty() + var login: Boolean by loginProperty + + val usernameProperty = SimpleStringProperty() + var username: String by usernameProperty + + val passwordProperty = SimpleStringProperty() + var password: String by passwordProperty + + val thanksProperty = SimpleBooleanProperty() + var thanks: Boolean by thanksProperty + + val hostProperty = SimpleStringProperty() + var host: String by hostProperty +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/AppView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/AppView.kt new file mode 100644 index 00000000..3eb9eb5b --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/AppView.kt @@ -0,0 +1,17 @@ +package me.mnlr.vripper.view + +import me.mnlr.vripper.view.actionbar.ActionBarView +import me.mnlr.vripper.view.main.MainView +import me.mnlr.vripper.view.status.StatusBarView +import tornadofx.View +import tornadofx.borderpane + +class AppView : View("Vripper") { + + override val root = borderpane { + top() + center() + bottom() + } + +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/LoadingView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/LoadingView.kt new file mode 100644 index 00000000..338ddacd --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/LoadingView.kt @@ -0,0 +1,30 @@ +package me.mnlr.vripper.view + +import javafx.scene.control.ProgressIndicator.INDETERMINATE_PROGRESS +import javafx.scene.effect.DropShadow +import javafx.util.Duration +import me.mnlr.vripper.event.ApplicationInitialized +import tornadofx.* + +class LoadingView : View("Vripper") { + + private val appView: AppView by inject() + + init { + subscribe { + runLater { + replaceWith(appView, ViewTransition.FadeThrough(Duration.millis(250.0))) + } + } + } + + override val root = borderpane { + padding = insets(all = 5) + center { + progressindicator { + progress = INDETERMINATE_PROGRESS + } + } + effect = DropShadow() + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/ActionBarView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/ActionBarView.kt new file mode 100644 index 00000000..5ee57840 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/ActionBarView.kt @@ -0,0 +1,29 @@ +package me.mnlr.vripper.view.actionbar + +import javafx.scene.control.ContentDisplay +import me.mnlr.vripper.gui.Styles +import me.mnlr.vripper.view.settings.SettingsView +import tornadofx.* + +class ActionBarView : View("Vripper") { + + override val root = borderpane { + padding = insets(all = 5) + left() + center() + right { + button("Settings") { + imageview("settings.png") { + fitWidth = 32.0 + fitHeight = 32.0 + } + addClass(Styles.actionBarButton) + contentDisplay = ContentDisplay.GRAPHIC_ONLY + tooltip("Open settings menu") + action { + find().openModal() + } + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/AddActionsView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/AddActionsView.kt new file mode 100644 index 00000000..4034d635 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/AddActionsView.kt @@ -0,0 +1,31 @@ +package me.mnlr.vripper.view.actionbar + +import javafx.geometry.Orientation +import javafx.scene.control.ContentDisplay +import javafx.scene.input.KeyCode +import javafx.scene.input.KeyCodeCombination +import javafx.scene.input.KeyCombination +import me.mnlr.vripper.gui.Styles +import me.mnlr.vripper.view.posts.AddView +import tornadofx.* + +class AddActionsView : View() { + + override val root = hbox { + button("Add links") { + imageview("plus.png") { + fitWidth = 32.0 + fitHeight = 32.0 + } + addClass(Styles.actionBarButton) + contentDisplay = ContentDisplay.GRAPHIC_ONLY + tooltip("Add new links [Ctrl+L]") + shortcut(KeyCodeCombination(KeyCode.L, KeyCombination.CONTROL_DOWN)) + action { + find().input.clear() + find().openModal() + } + } + separator(Orientation.VERTICAL) + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/DownloadActionsView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/DownloadActionsView.kt new file mode 100644 index 00000000..2668b595 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/DownloadActionsView.kt @@ -0,0 +1,134 @@ +package me.mnlr.vripper.view.actionbar + +import javafx.beans.property.SimpleBooleanProperty +import javafx.geometry.Orientation +import javafx.scene.control.ButtonType +import javafx.scene.control.ContentDisplay +import javafx.scene.input.KeyCode +import javafx.scene.input.KeyCodeCombination +import javafx.scene.input.KeyCombination +import me.mnlr.vripper.controller.GlobalStateController +import me.mnlr.vripper.controller.PostController +import me.mnlr.vripper.gui.Styles +import me.mnlr.vripper.view.tables.PostsTableView +import tornadofx.* + +class DownloadActionsView : View() { + + private val postController: PostController by inject() + private val globalStateController: GlobalStateController by inject() + private val postsTableView: PostsTableView by inject() + private val downloadActiveProperty = SimpleBooleanProperty(true) + + init { + downloadActiveProperty.bind(globalStateController.globalState.runningProperty.greaterThan(0)) + } + + override val root = hbox { + button("Start All") { + imageview("end.png") { + fitWidth = 32.0 + fitHeight = 32.0 + } + addClass(Styles.actionBarButton) + contentDisplay = ContentDisplay.GRAPHIC_ONLY + tooltip("Start downloads [Ctrl+S]") + shortcut(KeyCodeCombination(KeyCode.S, KeyCombination.CONTROL_DOWN)) + disableWhen(downloadActiveProperty) + action { + postController.startAll() + } + } + button("Stop All") { + setPrefSize(32.0, 32.0) + imageview("stop.png") { + fitWidth = 32.0 + fitHeight = 32.0 + } + addClass(Styles.actionBarButton) + contentDisplay = ContentDisplay.GRAPHIC_ONLY + tooltip("Stops all running downloads [Ctrl+Q]") + shortcut(KeyCodeCombination(KeyCode.Q, KeyCombination.CONTROL_DOWN)) + disableWhen(downloadActiveProperty.not()) + action { + postController.stopAll() + } + } + separator(Orientation.VERTICAL) + button("Start") { + setPrefSize(32.0, 32.0) + imageview("play.png") { + fitWidth = 32.0 + fitHeight = 32.0 + } + addClass(Styles.actionBarButton) + contentDisplay = ContentDisplay.GRAPHIC_ONLY + tooltip("Start downloads for selected [Ctrl+Shift+S]") + shortcut(KeyCodeCombination(KeyCode.S, KeyCombination.CONTROL_DOWN, KeyCombination.SHIFT_DOWN)) + enableWhen( + postsTableView.tableView.selectionModel.selectedItems.sizeProperty.greaterThan( + 0 + ) + ) + action { + postsTableView.startSelected() + } + } + button("Stop") { + setPrefSize(32.0, 32.0) + imageview("pause.png") { + fitWidth = 32.0 + fitHeight = 32.0 + } + addClass(Styles.actionBarButton) + contentDisplay = ContentDisplay.GRAPHIC_ONLY + tooltip("Stop downloads for selected [Ctrl+Shift+Q]") + shortcut(KeyCodeCombination(KeyCode.Q, KeyCombination.CONTROL_DOWN, KeyCombination.SHIFT_DOWN)) + enableWhen( + postsTableView.tableView.selectionModel.selectedItems.sizeProperty.greaterThan( + 0 + ) + ) + action { + postsTableView.stopSelected() + } + } + button("Delete") { + setPrefSize(32.0, 32.0) + imageview("trash.png") { + fitWidth = 32.0 + fitHeight = 32.0 + } + addClass(Styles.actionBarButton) + contentDisplay = ContentDisplay.GRAPHIC_ONLY + tooltip("Delete selected posts [Del]") + shortcut(KeyCodeCombination(KeyCode.DELETE)) + enableWhen( + postsTableView.tableView.selectionModel.selectedItems.sizeProperty.greaterThan( + 0 + ) + ) + action { + postsTableView.deleteSelected() + } + } + separator(Orientation.VERTICAL) + button("Clear") { + setPrefSize(32.0, 32.0) + imageview("broom.png") { + fitWidth = 32.0 + fitHeight = 32.0 + } + addClass(Styles.actionBarButton) + contentDisplay = ContentDisplay.GRAPHIC_ONLY + tooltip("Clear all finished downloads [Ctrl+Del]") + shortcut(KeyCodeCombination(KeyCode.DELETE, KeyCombination.CONTROL_DOWN)) + action { + confirm("Clean finished posts", "Confirm removal of finished posts", ButtonType.YES, ButtonType.NO) { + val clearPosts = postController.clearPosts() + postsTableView.tableView.items.removeIf { clearPosts.contains(it.postId) } + } + } + } + } +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/LogActionsView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/LogActionsView.kt new file mode 100644 index 00000000..bc73adcc --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/LogActionsView.kt @@ -0,0 +1,36 @@ +package me.mnlr.vripper.view.actionbar + +import javafx.scene.control.ButtonType +import javafx.scene.control.ContentDisplay +import javafx.scene.input.KeyCode +import javafx.scene.input.KeyCodeCombination +import javafx.scene.input.KeyCombination +import me.mnlr.vripper.controller.LogController +import me.mnlr.vripper.gui.Styles +import me.mnlr.vripper.view.tables.LogTableView +import tornadofx.* + +class LogActionsView : View() { + + private val logController: LogController by inject() + private val logTableView: LogTableView by inject() + + override val root = hbox { + button("Clear") { + imageview("broom.png") { + fitWidth = 32.0 + fitHeight = 32.0 + } + addClass(Styles.actionBarButton) + contentDisplay = ContentDisplay.GRAPHIC_ONLY + tooltip("Clear logs [Ctrl+Del]") + shortcut(KeyCodeCombination(KeyCode.DELETE, KeyCombination.CONTROL_DOWN)) + action { + confirm("Clear logs", "Are you sure you want to clear the logs", ButtonType.YES, ButtonType.NO) { + logController.clear() + logTableView.items.clear() + } + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/ThreadActionsView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/ThreadActionsView.kt new file mode 100644 index 00000000..65131be9 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/actionbar/ThreadActionsView.kt @@ -0,0 +1,54 @@ +package me.mnlr.vripper.view.actionbar + +import javafx.scene.control.ButtonType +import javafx.scene.control.ContentDisplay +import javafx.scene.input.KeyCode +import javafx.scene.input.KeyCodeCombination +import javafx.scene.input.KeyCombination +import me.mnlr.vripper.controller.ThreadController +import me.mnlr.vripper.gui.Styles +import me.mnlr.vripper.view.tables.ThreadTableView +import tornadofx.* + +class ThreadActionsView : View() { + + private val threadTableView: ThreadTableView by inject() + private val threadController: ThreadController by inject() + + override val root = hbox { + button("Delete") { + imageview("trash.png") { + fitWidth = 32.0 + fitHeight = 32.0 + } + addClass(Styles.actionBarButton) + contentDisplay = ContentDisplay.GRAPHIC_ONLY + tooltip("Delete selected threads [Del]") + shortcut(KeyCodeCombination(KeyCode.DELETE)) + enableWhen( + threadTableView.tableView.selectionModel.selectedItems.sizeProperty.greaterThan( + 0 + ) + ) + action { + threadTableView.deleteSelected() + } + } + button("Clear") { + imageview("broom.png") { + fitWidth = 32.0 + fitHeight = 32.0 + } + addClass(Styles.actionBarButton) + contentDisplay = ContentDisplay.GRAPHIC_ONLY + tooltip("Clear all threads [Ctrl+Del]") + shortcut(KeyCodeCombination(KeyCode.DELETE, KeyCombination.CONTROL_DOWN)) + action { + confirm("Clean threads", "Confirm removal of threads", ButtonType.YES, ButtonType.NO) { + threadController.clearAll() + threadTableView.tableView.items.clear() + } + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/main/MainView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/main/MainView.kt new file mode 100644 index 00000000..66d8a79c --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/main/MainView.kt @@ -0,0 +1,74 @@ +package me.mnlr.vripper.view.main + +import javafx.scene.control.TabPane +import javafx.scene.image.ImageView +import javafx.util.Duration +import me.mnlr.vripper.view.actionbar.ActionBarView +import me.mnlr.vripper.view.actionbar.DownloadActionsView +import me.mnlr.vripper.view.actionbar.LogActionsView +import me.mnlr.vripper.view.actionbar.ThreadActionsView +import me.mnlr.vripper.view.tables.LogTableView +import me.mnlr.vripper.view.tables.PostsTableView +import me.mnlr.vripper.view.tables.ThreadTableView +import tornadofx.* + +class MainView : View("Center view") { + + private val actionBarView: ActionBarView by inject() + + private val downloadActionBar: DownloadActionsView by inject() + private val threadActionsView: ThreadActionsView by inject() + private val logActionsView: LogActionsView by inject() + + override val root = tabpane { + tabClosingPolicy = TabPane.TabClosingPolicy.UNAVAILABLE + selectionModel.selectedItemProperty() + .addListener(ChangeListener { _, _, newValue -> + run { + when (newValue.id) { + "download-tab" -> { + actionBarView.root.center.replaceWith( + downloadActionBar.root, + ViewTransition.FadeThrough(Duration.millis(100.0)) + ) + } + + "thread-tab" -> { + actionBarView.root.center.replaceWith( + threadActionsView.root, + ViewTransition.FadeThrough(Duration.millis(100.0)) + ) + } + + "log-tab" -> { + actionBarView.root.center.replaceWith( + logActionsView.root, + ViewTransition.FadeThrough(Duration.millis(100.0)) + ) + } + } + } + }) + tab { + val imageView = ImageView("download.png") + imageView.fitWidth = 18.0 + imageView.fitHeight = 18.0 + id = "download-tab" + graphic = imageView + } + tab { + val imageView = ImageView("analyze.png") + imageView.fitWidth = 18.0 + imageView.fitHeight = 18.0 + id = "thread-tab" + graphic = imageView + } + tab { + val imageView = ImageView("bug.png") + imageView.fitWidth = 18.0 + imageView.fitHeight = 18.0 + id = "log-tab" + graphic = imageView + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/posts/AddView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/posts/AddView.kt new file mode 100644 index 00000000..2d22eae2 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/posts/AddView.kt @@ -0,0 +1,36 @@ +package me.mnlr.vripper.view.posts + +import javafx.beans.property.SimpleStringProperty +import javafx.geometry.Pos +import javafx.scene.control.TextArea +import javafx.scene.layout.Priority +import javafx.scene.layout.VBox +import me.mnlr.vripper.controller.PostController +import tornadofx.* + +class AddView : Fragment("Add thread links") { + + private val textAreaProperty = SimpleStringProperty() + private val postController: PostController by inject() + lateinit var input: TextArea + + override val root = vbox(alignment = Pos.CENTER_RIGHT) { + padding = insets(all = 5) + spacing = 5.0 + input = textarea { + VBox.setVgrow(this, Priority.ALWAYS) + bind(textAreaProperty) + } + button("Scan") { + imageview("search.png") { + fitWidth = 18.0 + fitHeight = 18.0 + } + disableWhen(textAreaProperty.isEmpty) + action { + postController.scan(textAreaProperty.value) + close() + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/ConnectionSettingsView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/ConnectionSettingsView.kt new file mode 100644 index 00000000..4107f47d --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/ConnectionSettingsView.kt @@ -0,0 +1,45 @@ +package me.mnlr.vripper.view.settings + +import me.mnlr.vripper.controller.SettingsController +import me.mnlr.vripper.model.settings.ConnectionSettingsModel +import tornadofx.* + +class ConnectionSettingsView : View("Connection settings") { + private val settingsController: SettingsController by inject() + val connectionSettingsModel = ConnectionSettingsModel() + + override fun onDock() { + val connectionSettings = settingsController.findConnectionSettings() + connectionSettingsModel.maxThreads = connectionSettings.maxThreads + connectionSettingsModel.maxTotalThreads = connectionSettings.maxTotalThreads + connectionSettingsModel.timeout = connectionSettings.timeout + connectionSettingsModel.maxAttempts = connectionSettings.maxAttempts + } + + override val root = vbox { + form { + fieldset { + field("Concurrent downloads per host") { + textfield(connectionSettingsModel.maxThreadsProperty) { + filterInput { it.controlNewText.isInt() } + } + } + field("Global concurrent downloads") { + textfield(connectionSettingsModel.maxTotalThreadsProperty) { + filterInput { it.controlNewText.isInt() } + } + } + field("Connection timeout") { + textfield(connectionSettingsModel.timeoutProperty) { + filterInput { it.controlNewText.isInt() } + } + } + field("Maximum attempts") { + textfield(connectionSettingsModel.maxAttemptsProperty) { + filterInput { it.controlNewText.isInt() } + } + } + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/DownloadSettingsView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/DownloadSettingsView.kt new file mode 100644 index 00000000..8c05ee2d --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/DownloadSettingsView.kt @@ -0,0 +1,79 @@ +package me.mnlr.vripper.view.settings + +import me.mnlr.vripper.controller.SettingsController +import me.mnlr.vripper.model.settings.DownloadSettingsModel +import tornadofx.* + + +class DownloadSettingsView : View("Download Settings") { + + private val settingsController: SettingsController by inject() + val downloadSettingsModel = DownloadSettingsModel() + + override fun onDock() { + val downloadSettings = settingsController.findDownloadSettings() + downloadSettingsModel.downloadPath = downloadSettings.downloadPath + downloadSettingsModel.autoStart = downloadSettings.autoStart + downloadSettingsModel.autoQueueThreshold = downloadSettings.autoQueueThreshold + downloadSettingsModel.forceOrder = downloadSettings.forceOrder + downloadSettingsModel.forumSubfolder = downloadSettings.forumSubfolder + downloadSettingsModel.threadSubLocation = downloadSettings.threadSubLocation + downloadSettingsModel.clearCompleted = downloadSettings.clearCompleted + downloadSettingsModel.appendPostId = downloadSettings.appendPostId + } + + override val root = vbox { + form { + fieldset { + field("Download Path") { + textfield(downloadSettingsModel.downloadPathProperty) { + isEditable = false + } + button("Browse") { + action { + val directory = chooseDirectory(title = "Select download folder") + if(directory != null) { + downloadSettingsModel.downloadPathProperty.set(directory.path) + } + } + } + } + field("Auto start downloads") { + checkbox { + bind(downloadSettingsModel.autoStartProperty) + } + } + field("Auto queue thread if post count is below or equal to") { + textfield(downloadSettingsModel.autoQueueThresholdProperty) { + filterInput { it.controlNewText.isInt() } + } + } + field("Organize by category") { + checkbox { + bind(downloadSettingsModel.forumSubfolderProperty) + } + } + field("Organize by thread") { + checkbox { + bind(downloadSettingsModel.threadSubLocationProperty) + } + } + field("Order images") { + checkbox { + bind(downloadSettingsModel.forceOrderProperty) + } + } + field("Append post id to download folder") { + checkbox { + bind(downloadSettingsModel.appendPostIdProperty) + } + } + field("Clear Finished") { + checkbox { + bind(downloadSettingsModel.clearCompletedProperty) + } + } + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/SettingsView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/SettingsView.kt new file mode 100644 index 00000000..c9a3e79d --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/SettingsView.kt @@ -0,0 +1,71 @@ +package me.mnlr.vripper.view.settings + +import javafx.geometry.Pos +import javafx.scene.control.Alert +import javafx.scene.control.TabPane +import javafx.scene.image.ImageView +import javafx.scene.layout.Priority +import javafx.scene.layout.VBox +import me.mnlr.vripper.controller.SettingsController +import me.mnlr.vripper.exception.ValidationException +import tornadofx.* + + +class SettingsView : Fragment("Settings") { + + private val settingsController: SettingsController by inject() + + private val downloadSettingsView: DownloadSettingsView by inject() + private val connectionSettingsView: ConnectionSettingsView by inject() + private val viperSettingsView: ViperSettingsView by inject() + + override val root = vbox(alignment = Pos.CENTER_RIGHT) { + spacing = 5.0 + tabpane { + tabClosingPolicy = TabPane.TabClosingPolicy.UNAVAILABLE + VBox.setVgrow(this, Priority.ALWAYS) + tab() { + val imageView = ImageView("downloads-folder.png") + imageView.fitWidth = 18.0 + imageView.fitHeight = 18.0 + graphic = imageView + } + tab() { + val imageView = ImageView("data-transfer.png") + imageView.fitWidth = 18.0 + imageView.fitHeight = 18.0 + graphic = imageView + } + tab() { + val imageView = ImageView("icons/32x32.png") + imageView.fitWidth = 18.0 + imageView.fitHeight = 18.0 + graphic = imageView + } + } + borderpane { + right { + padding = insets(all = 5.0) + button("Save") { + imageview("save.png") { + fitWidth = 18.0 + fitHeight = 18.0 + } + action { + try { + settingsController.saveNewSettings( + downloadSettingsView.downloadSettingsModel, + connectionSettingsView.connectionSettingsModel, + viperSettingsView.viperSettingsModel + ) + close() + } catch (e: ValidationException) { + alert(Alert.AlertType.ERROR, "Invalid settings", e.message) + } + + } + } + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/ViperSettingsView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/ViperSettingsView.kt new file mode 100644 index 00000000..a9f18136 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/settings/ViperSettingsView.kt @@ -0,0 +1,52 @@ +package me.mnlr.vripper.view.settings + +import javafx.collections.FXCollections +import me.mnlr.vripper.controller.SettingsController +import me.mnlr.vripper.model.settings.ViperSettingsModel +import tornadofx.* + +class ViperSettingsView : View("Viper Settings") { + private val settingsController: SettingsController by inject() + val viperSettingsModel = ViperSettingsModel() + val proxies = FXCollections.observableArrayList() + + override fun onDock() { + proxies.clear() + proxies.addAll(settingsController.getProxies()) + val viperGirlsSettings = settingsController.findViperGirlsSettings() + viperSettingsModel.login = viperGirlsSettings.login + viperSettingsModel.username = viperGirlsSettings.username + viperSettingsModel.password = viperGirlsSettings.password + viperSettingsModel.thanks = viperGirlsSettings.thanks + viperSettingsModel.host = viperGirlsSettings.host + } + + override val root = vbox { + form { + fieldset { + field("Enable ViperGirls Authentication") { + checkbox { + bind(viperSettingsModel.loginProperty) + } + } + fieldset { + visibleWhen(viperSettingsModel.loginProperty) + field("ViperGirls Username") { + textfield(viperSettingsModel.usernameProperty) + } + field("ViperGirls Password") { + passwordfield(viperSettingsModel.passwordProperty) + } + field("Leave like") { + checkbox { + bind(viperSettingsModel.thanksProperty) + } + } + field("Select a proxy") { + combobox(viperSettingsModel.hostProperty, proxies) + } + } + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/status/StatusBarView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/status/StatusBarView.kt new file mode 100644 index 00000000..2e1445bc --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/status/StatusBarView.kt @@ -0,0 +1,36 @@ +package me.mnlr.vripper.view.status + +import javafx.geometry.Orientation +import javafx.geometry.Pos +import me.mnlr.vripper.controller.GlobalStateController +import tornadofx.* + +class StatusBarView : View("Status bar") { + + private val globalStateController: GlobalStateController by inject() + + override val root = borderpane { + left { + text(globalStateController.globalState.loggedUserProperty.map { "Logged in as: $it" }) { + visibleWhen(globalStateController.globalState.loggedUserProperty.isNotBlank()) + } + } + right { + padding = insets(right = 5, left = 5, top = 3, bottom = 3) + hbox { + spacing = 3.0 + text(globalStateController.globalState.downloadSpeedProperty.map { "$it/s" }) + separator(Orientation.VERTICAL) + label("Downloading") + text(globalStateController.globalState.runningProperty.asString()) + separator(Orientation.VERTICAL) + label("Pending") + text(globalStateController.globalState.remainingProperty.asString()) + separator(Orientation.VERTICAL) + label("Error") + text(globalStateController.globalState.errorProperty.asString()) + alignment = Pos.CENTER_RIGHT + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/ImagesTableView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/ImagesTableView.kt new file mode 100644 index 00000000..bd1002b2 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/ImagesTableView.kt @@ -0,0 +1,97 @@ +package me.mnlr.vripper.view.tables + +import javafx.collections.FXCollections +import javafx.collections.ObservableList +import javafx.geometry.Pos +import javafx.scene.control.TableView +import javafx.scene.input.MouseButton +import me.mnlr.vripper.controller.ImageController +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.model.ImageModel +import tornadofx.* + +class ImagesTableView : Fragment("Photos") { + + private lateinit var tableView: TableView + private val imageController: ImageController by inject() + private val eventBus: EventBus by di() + private var items: ObservableList = FXCollections.observableArrayList() + val postId: String by param() + + override fun onDock() { + tableView.prefWidthProperty().bind(root.widthProperty()) + tableView.prefHeightProperty().bind(root.heightProperty()) + modalStage?.width = 600.0 + + runLater { + items.addAll(imageController.findImages(postId)) + tableView.sort() + + val disposable = eventBus.flux() + .filter { it!!.kind == Event.Kind.IMAGE_UPDATE } + .subscribe { event -> + imageController + .findImageById(event!!.data as Long) + .filter { it.postId == postId } + .ifPresent { + // search + val find = items + .find { image -> image.id == it.id } + if (find != null) { + find.apply { + progress = it.progress + status = it.status + } + } else { + items.add(it) + } + } + } + + whenUndocked { + disposable.dispose() + } + } + } + + override val root = vbox(alignment = Pos.CENTER_RIGHT) { + tableView = tableview(items) { + column("Index", ImageModel::indexProperty) { + sortOrder.add(this) + } + column("Link", ImageModel::urlProperty) { + prefWidth = 200.0 + } + column("Progress", ImageModel::progressProperty) { + cellFormat { + addClass(Stylesheet.progressBarTableCell) + graphic = cache { + progressbar(itemProperty().doubleBinding { it?.toDouble() ?: 0.0 }) { + setOnMouseClicked { + when (it.clickCount) { + 1 -> { + this@tableview.requestFocus() + this@tableview.focusModel.focus(this@cellFormat.tableRow.index) + if (it.isControlDown && it.button.equals(MouseButton.PRIMARY)) { + if (this@tableview.selectionModel.isSelected(this@cellFormat.tableRow.index)) { + this@tableview.selectionModel.clearSelection(this@cellFormat.tableRow.index) + } else { + this@tableview.selectionModel.select(this@cellFormat.tableRow.index) + } + } else if (it.button.equals(MouseButton.PRIMARY)) { + this@tableview.selectionModel.clearSelection() + this@tableview.selectionModel.select(this@cellFormat.tableRow.index) + } + } + } + } + useMaxWidth = true + } + } + } + } + column("Status", ImageModel::statusProperty) + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/LogTableView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/LogTableView.kt new file mode 100644 index 00000000..9974a7ae --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/LogTableView.kt @@ -0,0 +1,64 @@ +package me.mnlr.vripper.view.tables + +import javafx.collections.FXCollections +import javafx.collections.ObservableList +import javafx.scene.control.SelectionMode +import javafx.scene.control.TableView +import me.mnlr.vripper.controller.LogController +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.model.LogModel +import tornadofx.* + +class LogTableView : View("Log") { + + private val logController: LogController by inject() + private val eventBus: EventBus by di() + + lateinit var tableView: TableView + + var items: ObservableList = FXCollections.observableArrayList() + + init { + items.addAll(logController.findAll()) + + eventBus.flux() + .doOnNext { event -> + if (event!!.kind.equals(Event.Kind.LOG_EVENT_UPDATE)) { + logController.find(event.data as Long) + .ifPresent { + // search + val find = items + .find { threadModel -> threadModel.id == it.id } + if (find != null) { + find.apply { + status = it.status + message = it.message + } + } else { + items.add(it) + } + } + } else if (event.kind.equals(Event.Kind.LOG_EVENT_REMOVE)) { + items.removeIf { it.id == event.data } + } + } + .subscribe() + } + + override fun onDock() { + tableView.prefHeightProperty().bind(root.heightProperty()) + } + + override val root = vbox { + tableView = tableview(items) { + selectionModel.selectionMode = SelectionMode.MULTIPLE + column("Time", LogModel::timeProperty) { + sortOrder.add(this) + } + column("Type", LogModel::typeProperty) + column("Status", LogModel::statusProperty) + column("Message", LogModel::messageProperty) + } + } +} diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/PostsTableView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/PostsTableView.kt new file mode 100644 index 00000000..d95d1a13 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/PostsTableView.kt @@ -0,0 +1,178 @@ +package me.mnlr.vripper.view.tables + +import javafx.collections.FXCollections +import javafx.collections.ObservableList +import javafx.scene.control.* +import javafx.scene.image.ImageView +import javafx.scene.input.MouseButton +import me.mnlr.vripper.controller.PostController +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.model.PostModel +import tornadofx.* + + +class PostsTableView : View("Download") { + + private val postController: PostController by inject() + private val eventBus: EventBus by di() + + lateinit var tableView: TableView + private var items: ObservableList = FXCollections.observableArrayList() + + init { + items.addAll(postController.findAllPosts()) + + eventBus.flux().filter { + it!!.kind == Event.Kind.POST_UPDATE || it.kind == Event.Kind.METADATA_UPDATE + }.subscribe { event -> + postController.findById(event!!.data as Long).ifPresent { + // search + val find = items.find { postModel -> postModel.postId == it.postId } + if (find != null) { + find.apply { + progress = it.progress + status = it.status + done = it.done + total = it.total + order = it.order + } + } else { + items.add(it) + runLater { + this.tableView.refresh() + } + } + } + + } + } + + override fun onDock() { + tableView.prefHeightProperty().bind(root.heightProperty()) + } + + override val root = vbox { + tableView = tableview(items) { + selectionModel.selectionMode = SelectionMode.MULTIPLE + setRowFactory { + val tableRow = TableRow() + tableRow.setOnMouseClicked { + if (it.clickCount == 2 && tableRow.item != null) { + openPhotos(tableRow.item.postId) + } + } + + val contextMenu = ContextMenu() + val startItem = MenuItem("Start") + startItem.setOnAction { + startSelected() + } + val playIcon = ImageView("play.png") + playIcon.fitWidth = 18.0 + playIcon.fitHeight = 18.0 + startItem.graphic = playIcon + + val stopItem = MenuItem("Stop") + stopItem.setOnAction { + stopSelected() + } + val stopIcon = ImageView("pause.png") + stopIcon.fitWidth = 18.0 + stopIcon.fitHeight = 18.0 + stopItem.graphic = stopIcon + + val deleteItem = MenuItem("Delete") + deleteItem.setOnAction { + deleteSelected() + } + val deleteIcon = ImageView("trash.png") + deleteIcon.fitWidth = 18.0 + deleteIcon.fitHeight = 18.0 + deleteItem.graphic = deleteIcon + + val detailsItem = MenuItem("Images") + detailsItem.setOnAction { + openPhotos(tableRow.item.postId) + } + val detailsIcon = ImageView("details.png") + detailsIcon.fitWidth = 18.0 + detailsIcon.fitHeight = 18.0 + detailsItem.graphic = detailsIcon + + contextMenu.items.addAll( + startItem, stopItem, deleteItem, SeparatorMenuItem(), detailsItem + ) + tableRow.contextMenuProperty() + .bind(tableRow.emptyProperty().map { empty -> if (empty) null else contextMenu }) + tableRow + } + column("Title", PostModel::titleProperty) { + prefWidth = 300.0 + } + column("Progress", PostModel::progressProperty) { + cellFormat { + addClass(Stylesheet.progressBarTableCell) + graphic = cache { + progressbar(itemProperty().doubleBinding { it?.toDouble() ?: 0.0 }) { + setOnMouseClicked { + when (it.clickCount) { + 1 -> { + this@tableview.requestFocus() + this@tableview.focusModel.focus(this@cellFormat.tableRow.index) + if (it.isControlDown && it.button.equals(MouseButton.PRIMARY)) { + if (this@tableview.selectionModel.isSelected(this@cellFormat.tableRow.index)) { + this@tableview.selectionModel.clearSelection(this@cellFormat.tableRow.index) + } else { + this@tableview.selectionModel.select(this@cellFormat.tableRow.index) + } + } else if (it.button.equals(MouseButton.PRIMARY)) { + this@tableview.selectionModel.clearSelection() + this@tableview.selectionModel.select(this@cellFormat.tableRow.index) + } + } + 2 -> openPhotos(this@cellFormat.tableRow.item.postId) + } + } + useMaxWidth = true + } + } + } + } + column("Status", PostModel::statusProperty) + column("Path", PostModel::pathProperty) + column("Images", PostModel::totalProperty) + column("Added On", PostModel::addedOnProperty) + column("Order", PostModel::orderProperty) { + sortOrder.add(this) + } + } + } + + fun deleteSelected() { + val postIdList = tableView.selectionModel.selectedItems.map { it.postId } + confirm( + "Remove posts", + "Confirm removal of ${postIdList.size} post${if (postIdList.size > 1) "s" else ""}", + ButtonType.YES, + ButtonType.NO + ) { + postController.delete(postIdList) + tableView.items.removeIf { postIdList.contains(it.postId) } + } + } + + fun stopSelected() { + val postIdList = tableView.selectionModel.selectedItems.map { it.postId } + postController.stop(postIdList) + } + + fun startSelected() { + val postIdList = tableView.selectionModel.selectedItems.map { it.postId } + postController.start(postIdList) + } + + private fun openPhotos(postId: String) { + find(mapOf(ImagesTableView::postId to postId)).openModal() + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/ThreadSelectionTableView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/ThreadSelectionTableView.kt new file mode 100644 index 00000000..6c9e27d9 --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/ThreadSelectionTableView.kt @@ -0,0 +1,73 @@ +package me.mnlr.vripper.view.tables + +import javafx.collections.FXCollections +import javafx.collections.ObservableList +import javafx.geometry.Pos +import javafx.scene.control.Label +import javafx.scene.control.SelectionMode +import javafx.scene.control.TableRow +import javafx.scene.control.TableView +import me.mnlr.vripper.controller.ThreadController +import me.mnlr.vripper.model.ThreadSelectionModel +import tornadofx.* + +class ThreadSelectionTableView : Fragment("Thread") { + + private lateinit var tableView: TableView + private val threadController: ThreadController by inject() + private var items: ObservableList = FXCollections.observableArrayList() + val threadId: String by param() + + override fun onDock() { + tableView.prefWidthProperty().bind(root.widthProperty()) + tableView.prefHeightProperty().bind(root.heightProperty()) + modalStage?.width = 600.0 + tableView.placeholder = Label("Loading") + + runLater { + items.addAll(threadController.grab(threadId)) + } + } + + override val root = vbox(alignment = Pos.CENTER_RIGHT) { + spacing = 5.0 + tableView = tableview(items) { + selectionModel.selectionMode = SelectionMode.MULTIPLE + setRowFactory { + val tableRow = TableRow() + + tableRow.setOnMouseClicked { + if (it.clickCount == 2 && tableRow.item != null) { + threadController.download(listOf(tableRow.item)) + close() + } + } + + tableRow + } + column("Post Index", ThreadSelectionModel::indexProperty) { + sortOrder.add(this) + } + column("Title", ThreadSelectionModel::titleProperty) { prefWidth = 200.0 } + column("URL", ThreadSelectionModel::urlProperty) { prefWidth = 200.0 } + column("Hosts", ThreadSelectionModel::hostsProperty) + } + borderpane { + right { + padding = insets(top = 0, right = 5, bottom = 5, left = 5) + button("Download") { + imageview("download.png") { + fitWidth = 18.0 + fitHeight = 18.0 + } + tooltip("Download selected posts") + enableWhen { tableView.selectionModel.selectedItems.sizeProperty.greaterThan(0) } + action { + threadController.download(tableView.selectionModel.selectedItems) + close() + } + } + } + } + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/ThreadTableView.kt b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/ThreadTableView.kt new file mode 100644 index 00000000..9d37dd9c --- /dev/null +++ b/vripper-gui/src/main/kotlin/me/mnlr/vripper/view/tables/ThreadTableView.kt @@ -0,0 +1,110 @@ +package me.mnlr.vripper.view.tables + +import javafx.collections.FXCollections +import javafx.collections.ObservableList +import javafx.scene.control.* +import javafx.scene.image.ImageView +import me.mnlr.vripper.controller.ThreadController +import me.mnlr.vripper.event.Event +import me.mnlr.vripper.event.EventBus +import me.mnlr.vripper.model.ThreadModel +import me.mnlr.vripper.view.main.MainView +import tornadofx.* + +class ThreadTableView : View("Threads") { + + private val threadController: ThreadController by inject() + private val eventBus: EventBus by di() + + lateinit var tableView: TableView + + private var items: ObservableList = FXCollections.observableArrayList() + + init { + items.addAll(threadController.findAll()) + + eventBus.flux().filter { it!!.kind == Event.Kind.THREAD_UPDATE }.subscribe { event -> + threadController.find(event!!.data as Long).ifPresent { + // search + val find = items.find { threadModel -> threadModel.threadId == it.threadId } + if (find != null) { + find.apply { + total = it.total + } + } else { + items.add(it) + runLater { + find().root.selectionModel.select(1) + tableView.selectionModel.clearSelection() + tableView.selectionModel.select(it) + } + } + } + } + } + + override fun onDock() { + tableView.prefHeightProperty().bind(root.heightProperty()) + } + + override val root = vbox { + tableView = tableview(items) { + selectionModel.selectionMode = SelectionMode.MULTIPLE + setRowFactory { + val tableRow = TableRow() + + tableRow.setOnMouseClicked { + if (it.clickCount == 2 && tableRow.item != null) { + selectPosts(tableRow.item.threadId) + } + } + + val contextMenu = ContextMenu() + val selectItem = MenuItem("Select posts") + selectItem.setOnAction { + selectPosts(tableRow.item.threadId) + } + val selectIcon = ImageView("popup.png") + selectIcon.fitWidth = 18.0 + selectIcon.fitHeight = 18.0 + selectItem.graphic = selectIcon + + val deleteItem = MenuItem("Delete") + deleteItem.setOnAction { + deleteSelected() + } + val deleteIcon = ImageView("trash.png") + deleteIcon.fitWidth = 18.0 + deleteIcon.fitHeight = 18.0 + deleteItem.graphic = deleteIcon + + contextMenu.items.addAll(selectItem, SeparatorMenuItem(), deleteItem) + tableRow.contextMenuProperty() + .bind(tableRow.emptyProperty().map { empty -> if (empty) null else contextMenu }) + tableRow + } + column("URL", ThreadModel::linkProperty) { + prefWidth = 350.0 + } + column("Count", ThreadModel::totalProperty) + } + } + + fun deleteSelected() { + val threadIdList = tableView.selectionModel.selectedItems.map { it.threadId } + confirm( + "Clean threads", + "Confirm removal of ${threadIdList.size} thread${if (threadIdList.size > 1) "s" else ""}", + ButtonType.YES, + ButtonType.NO + ) { + threadController.delete(threadIdList) + tableView.items.removeIf { threadIdList.contains(it.threadId) } + } + } + + private fun selectPosts(threadId: String) { + find(mapOf(ThreadSelectionTableView::threadId to threadId)).openModal() + + } +} \ No newline at end of file diff --git a/vripper-gui/src/main/resources/analyze.png b/vripper-gui/src/main/resources/analyze.png new file mode 100644 index 00000000..48b24106 Binary files /dev/null and b/vripper-gui/src/main/resources/analyze.png differ diff --git a/vripper-gui/src/main/resources/broom.png b/vripper-gui/src/main/resources/broom.png new file mode 100644 index 00000000..f31ba223 Binary files /dev/null and b/vripper-gui/src/main/resources/broom.png differ diff --git a/vripper-gui/src/main/resources/bug.png b/vripper-gui/src/main/resources/bug.png new file mode 100644 index 00000000..3de697f0 Binary files /dev/null and b/vripper-gui/src/main/resources/bug.png differ diff --git a/vripper-gui/src/main/resources/data-transfer.png b/vripper-gui/src/main/resources/data-transfer.png new file mode 100644 index 00000000..42f19635 Binary files /dev/null and b/vripper-gui/src/main/resources/data-transfer.png differ diff --git a/vripper-gui/src/main/resources/details.png b/vripper-gui/src/main/resources/details.png new file mode 100644 index 00000000..edd7cf0b Binary files /dev/null and b/vripper-gui/src/main/resources/details.png differ diff --git a/vripper-gui/src/main/resources/download.png b/vripper-gui/src/main/resources/download.png new file mode 100644 index 00000000..08ce0855 Binary files /dev/null and b/vripper-gui/src/main/resources/download.png differ diff --git a/vripper-gui/src/main/resources/downloads-folder.png b/vripper-gui/src/main/resources/downloads-folder.png new file mode 100644 index 00000000..ed5bd2d9 Binary files /dev/null and b/vripper-gui/src/main/resources/downloads-folder.png differ diff --git a/vripper-gui/src/main/resources/end.png b/vripper-gui/src/main/resources/end.png new file mode 100644 index 00000000..206dc95b Binary files /dev/null and b/vripper-gui/src/main/resources/end.png differ diff --git a/vripper-gui/src/main/resources/icons/1024x1024.png b/vripper-gui/src/main/resources/icons/1024x1024.png new file mode 100644 index 00000000..aac3a2f3 Binary files /dev/null and b/vripper-gui/src/main/resources/icons/1024x1024.png differ diff --git a/vripper-gui/src/main/resources/icons/128x128.png b/vripper-gui/src/main/resources/icons/128x128.png new file mode 100644 index 00000000..e6db8fa9 Binary files /dev/null and b/vripper-gui/src/main/resources/icons/128x128.png differ diff --git a/vripper-gui/src/main/resources/icons/16x16.png b/vripper-gui/src/main/resources/icons/16x16.png new file mode 100644 index 00000000..878807a1 Binary files /dev/null and b/vripper-gui/src/main/resources/icons/16x16.png differ diff --git a/vripper-gui/src/main/resources/icons/256x256.png b/vripper-gui/src/main/resources/icons/256x256.png new file mode 100644 index 00000000..756f8ef6 Binary files /dev/null and b/vripper-gui/src/main/resources/icons/256x256.png differ diff --git a/vripper-gui/src/main/resources/icons/32x32.png b/vripper-gui/src/main/resources/icons/32x32.png new file mode 100644 index 00000000..9d0f89e7 Binary files /dev/null and b/vripper-gui/src/main/resources/icons/32x32.png differ diff --git a/vripper-gui/src/main/resources/icons/48x48.png b/vripper-gui/src/main/resources/icons/48x48.png new file mode 100644 index 00000000..9a35c256 Binary files /dev/null and b/vripper-gui/src/main/resources/icons/48x48.png differ diff --git a/vripper-gui/src/main/resources/icons/512x512.png b/vripper-gui/src/main/resources/icons/512x512.png new file mode 100644 index 00000000..7f8b69f8 Binary files /dev/null and b/vripper-gui/src/main/resources/icons/512x512.png differ diff --git a/vripper-gui/src/main/resources/icons/64x64.png b/vripper-gui/src/main/resources/icons/64x64.png new file mode 100644 index 00000000..98f7dcb3 Binary files /dev/null and b/vripper-gui/src/main/resources/icons/64x64.png differ diff --git a/vripper-gui/src/main/resources/pause.png b/vripper-gui/src/main/resources/pause.png new file mode 100644 index 00000000..0f94e870 Binary files /dev/null and b/vripper-gui/src/main/resources/pause.png differ diff --git a/vripper-gui/src/main/resources/play.png b/vripper-gui/src/main/resources/play.png new file mode 100644 index 00000000..055a1256 Binary files /dev/null and b/vripper-gui/src/main/resources/play.png differ diff --git a/vripper-gui/src/main/resources/plus.png b/vripper-gui/src/main/resources/plus.png new file mode 100644 index 00000000..01ae16c1 Binary files /dev/null and b/vripper-gui/src/main/resources/plus.png differ diff --git a/vripper-gui/src/main/resources/popup.png b/vripper-gui/src/main/resources/popup.png new file mode 100644 index 00000000..67d53394 Binary files /dev/null and b/vripper-gui/src/main/resources/popup.png differ diff --git a/vripper-gui/src/main/resources/rename.png b/vripper-gui/src/main/resources/rename.png new file mode 100644 index 00000000..fec27f74 Binary files /dev/null and b/vripper-gui/src/main/resources/rename.png differ diff --git a/vripper-gui/src/main/resources/save.png b/vripper-gui/src/main/resources/save.png new file mode 100644 index 00000000..07d83e06 Binary files /dev/null and b/vripper-gui/src/main/resources/save.png differ diff --git a/vripper-gui/src/main/resources/search.png b/vripper-gui/src/main/resources/search.png new file mode 100644 index 00000000..26c5807e Binary files /dev/null and b/vripper-gui/src/main/resources/search.png differ diff --git a/vripper-gui/src/main/resources/settings.png b/vripper-gui/src/main/resources/settings.png new file mode 100644 index 00000000..bcbedf42 Binary files /dev/null and b/vripper-gui/src/main/resources/settings.png differ diff --git a/vripper-gui/src/main/resources/stop.png b/vripper-gui/src/main/resources/stop.png new file mode 100644 index 00000000..f6c8a5e8 Binary files /dev/null and b/vripper-gui/src/main/resources/stop.png differ diff --git a/vripper-gui/src/main/resources/trash.png b/vripper-gui/src/main/resources/trash.png new file mode 100644 index 00000000..b5dc52e4 Binary files /dev/null and b/vripper-gui/src/main/resources/trash.png differ diff --git a/vripper-server/pom.xml b/vripper-server/pom.xml deleted file mode 100644 index 4a9b7112..00000000 --- a/vripper-server/pom.xml +++ /dev/null @@ -1,160 +0,0 @@ - - - 4.0.0 - - tn.mnlr - vripper - 3.5.4 - - vripper-server - vripper-server - vripper-server - - - 11 - - - - - org.springframework.boot - spring-boot-starter-websocket - - - org.springframework.boot - spring-boot-starter-actuator - - - org.springframework.boot - spring-boot-starter-data-jdbc - - - org.liquibase - liquibase-core - - - org.hsqldb - hsqldb - runtime - - - org.projectlombok - lombok - true - - - org.springframework.boot - spring-boot-starter-test - test - - - org.springframework.restdocs - spring-restdocs-mockmvc - test - - - org.apache.httpcomponents - httpclient - - - net.sourceforge.htmlcleaner - htmlcleaner - 2.24 - - - io.projectreactor - reactor-core - - - net.jodah - failsafe - 2.0.1 - - - com.github.ben-manes.caffeine - caffeine - - - tn.mnlr - vripper-ui - ${project.version} - runtime - ${buildClassifier} - - - - - - - maven-jar-plugin - - ${buildClassifier} - - - ${maven.build.timestamp} - ${user.name} - - - - - - org.springframework.boot - spring-boot-maven-plugin - - - repackage - - repackage - - - ${buildClassifier} - - - - - - - - - - web - - true - - - web - - - - - maven-resources-plugin - - - copy-resources - validate - - copy-resources - - - ${project.build.directory}/classes/static - - - ${project.parent.basedir}/vripper-ui/dist/vripper-ui/ - - - - - - - - - - - - electron - - electron - - - - diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/EventListenerBean.java b/vripper-server/src/main/java/tn/mnlr/vripper/EventListenerBean.java deleted file mode 100644 index 423d58a6..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/EventListenerBean.java +++ /dev/null @@ -1,31 +0,0 @@ -package tn.mnlr.vripper; - -import lombok.Getter; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.context.event.ContextRefreshedEvent; -import org.springframework.context.event.EventListener; -import org.springframework.stereotype.Component; -import tn.mnlr.vripper.jpa.repositories.IPostRepository; -import tn.mnlr.vripper.services.DataService; - -@Component -public class EventListenerBean { - - @Getter private static boolean init = false; - - private final IPostRepository postRepository; - private final DataService dataService; - - @Autowired - public EventListenerBean(IPostRepository postRepository, DataService dataService) { - this.postRepository = postRepository; - this.dataService = dataService; - } - - @EventListener - public void onApplicationEvent(ContextRefreshedEvent event) { - postRepository.setDownloadingToStopped(); - dataService.sortPostsByRank(); - init = true; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/SpringContext.java b/vripper-server/src/main/java/tn/mnlr/vripper/SpringContext.java deleted file mode 100644 index 90034f80..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/SpringContext.java +++ /dev/null @@ -1,39 +0,0 @@ -package tn.mnlr.vripper; - -import lombok.extern.slf4j.Slf4j; -import org.springframework.beans.BeansException; -import org.springframework.context.ApplicationContext; -import org.springframework.context.ApplicationContextAware; -import org.springframework.context.ConfigurableApplicationContext; -import org.springframework.stereotype.Component; - -import java.util.Map; - -@Component -@Slf4j -public class SpringContext implements ApplicationContextAware { - - private static ConfigurableApplicationContext context; - - public static T getBean(Class beanClass) { - return context.getBean(beanClass); - } - - public static Map getBeansOfType(Class beanClass) { - return context.getBeansOfType(beanClass); - } - - public static void close() { - log.info("Application terminating..."); - if (context != null) { - context.close(); - } - } - - @Override - public void setApplicationContext(ApplicationContext context) throws BeansException { - - // store ApplicationContext reference to access required beans later on - SpringContext.context = ((ConfigurableApplicationContext) context); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/Utils.java b/vripper-server/src/main/java/tn/mnlr/vripper/Utils.java deleted file mode 100644 index b9ea31ef..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/Utils.java +++ /dev/null @@ -1,24 +0,0 @@ -package tn.mnlr.vripper; - -import lombok.extern.slf4j.Slf4j; - -import java.io.IOException; -import java.io.PrintWriter; -import java.io.StringWriter; - -@Slf4j -public class Utils { - - public static String throwableToString(Throwable th) { - try (StringWriter stringWriter = new StringWriter(); - PrintWriter printWriter = new PrintWriter(stringWriter)) { - th.printStackTrace(printWriter); - printWriter.flush(); - stringWriter.flush(); - return stringWriter.toString(); - } catch (IOException e) { - log.error("Failed to convert throwable to string", e); - } - return "Cannot display error, please check the log file for details"; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/VripperApplication.java b/vripper-server/src/main/java/tn/mnlr/vripper/VripperApplication.java deleted file mode 100644 index a1001823..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/VripperApplication.java +++ /dev/null @@ -1,19 +0,0 @@ -package tn.mnlr.vripper; - -import lombok.extern.slf4j.Slf4j; -import org.springframework.boot.SpringApplication; -import org.springframework.boot.autoconfigure.SpringBootApplication; - -@SpringBootApplication -@Slf4j -public class VripperApplication { - - public static void main(String[] args) { - try { - Runtime.getRuntime().addShutdownHook(new Thread(SpringContext::close)); - SpringApplication.run(VripperApplication.class, args); - } catch (Exception e) { - log.error("Failed to run the application", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/download/DownloadJob.java b/vripper-server/src/main/java/tn/mnlr/vripper/download/DownloadJob.java deleted file mode 100644 index d08e1bac..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/download/DownloadJob.java +++ /dev/null @@ -1,295 +0,0 @@ -package tn.mnlr.vripper.download; - -import lombok.Getter; -import lombok.extern.slf4j.Slf4j; -import net.jodah.failsafe.function.CheckedRunnable; -import org.apache.http.client.HttpClient; -import org.apache.http.client.methods.AbstractExecutionAwareRequest; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.client.protocol.HttpClientContext; -import org.apache.http.impl.client.BasicCookieStore; -import org.apache.http.util.EntityUtils; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.exception.DownloadException; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.jpa.domain.Image; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.jpa.domain.enums.Status; -import tn.mnlr.vripper.services.*; -import tn.mnlr.vripper.services.domain.Settings; - -import javax.imageio.ImageIO; -import javax.imageio.ImageReader; -import javax.imageio.stream.ImageInputStream; -import java.io.File; -import java.io.FileOutputStream; -import java.io.IOException; -import java.io.InputStream; -import java.nio.file.Files; -import java.nio.file.StandardCopyOption; -import java.util.*; - -@Slf4j -public class DownloadJob implements CheckedRunnable { - - private static final Object LOCK = new Object(); - private static final int READ_BUFFER_SIZE = 8192; - private final DataService dataService; - private final PathService pathService; - private final ConnectionService cm; - private final VGAuthService authService; - private final DownloadSpeedService downloadSpeedService; - private final Settings settings; - private final HttpClientContext context; - @Getter private final Image image; - @Getter private final Post post; - private volatile boolean stopped = false; - @Getter private volatile boolean finished = false; - - DownloadJob(Post post, Image image, Settings settings) { - this.image = image; - this.post = post; - this.settings = settings; - dataService = SpringContext.getBean(DataService.class); - pathService = SpringContext.getBean(PathService.class); - cm = SpringContext.getBean(ConnectionService.class); - authService = SpringContext.getBean(VGAuthService.class); - downloadSpeedService = SpringContext.getBean(DownloadSpeedService.class); - context = HttpClientContext.create(); - context.setCookieStore(new BasicCookieStore()); - context.setAttribute( - ContextAttributes.OPEN_CONNECTION.toString(), - Collections.synchronizedList(new ArrayList())); - } - - public void download(final Post post, final Image image) throws DownloadException { - - try { - - image.setStatus(Status.DOWNLOADING); - image.setCurrent(0); - dataService.updateImageStatus(image.getStatus(), image.getId()); - dataService.updateImageCurrent(image.getCurrent(), image.getId()); - - synchronized (LOCK) { - if (!post.getStatus().equals(Status.DOWNLOADING) - && !post.getStatus().equals(Status.PARTIAL)) { - post.setStatus(Status.DOWNLOADING); - dataService.updatePostStatus(post.getStatus(), post.getId()); - } - - // The post may be updated and the download directory might be set by another thread - Post updatedPost = dataService.findById(post.getId()).orElseThrow(); - if (updatedPost.getDownloadDirectory() == null) { - pathService.createDefaultPostFolder(updatedPost, settings); - } - if (settings.getLeaveThanksOnStart() != null && settings.getLeaveThanksOnStart()) { - authService.leaveThanks(updatedPost); - } - } - - if (stopped) { - return; - } - /* - * HOST SPECIFIC - */ - log.debug( - String.format( - "Getting image url and name from %s using %s", image.getUrl(), image.getHost())); - HostService.NameUrl nameAndUrl = image.getHost().getNameAndUrl(image.getUrl(), context); - log.debug(String.format("Resolved name for %s: %s", image.getUrl(), nameAndUrl.getName())); - log.debug( - String.format("Resolved image url for %s: %s", image.getUrl(), nameAndUrl.getUrl())); - /* - * END HOST SPECIFIC - */ - - if (stopped) { - return; - } - String formatImageFileName = pathService.formatImageFileName(nameAndUrl.getName()); - log.debug( - String.format( - "Sanitizing image name from %s to %s", nameAndUrl.getName(), formatImageFileName)); - nameAndUrl = new HostService.NameUrl(formatImageFileName, nameAndUrl.getUrl()); - - HttpClient client = cm.getClient().build(); - - log.debug(String.format("Downloading %s", nameAndUrl.getUrl())); - HttpGet httpGet = cm.buildHttpGet(nameAndUrl.getUrl(), context); - httpGet.addHeader("Referer", image.getUrl()); - try (CloseableHttpResponse response = - (CloseableHttpResponse) client.execute(httpGet, context)) { - - if (response.getStatusLine().getStatusCode() / 100 != 2) { - EntityUtils.consumeQuietly(response.getEntity()); - throw new DownloadException( - String.format("Server returned code %d", response.getStatusLine().getStatusCode())); - } - if (stopped) { - return; - } - File outputFile = Files.createTempFile("vripper", "tmp").toFile(); - try (InputStream downloadStream = response.getEntity().getContent(); - FileOutputStream fos = new FileOutputStream(outputFile)) { - - if (stopped) { - return; - } - - image.setTotal(response.getEntity().getContentLength()); - dataService.updateImageTotal(image.getTotal(), image.getId()); - - log.debug(String.format("%s length is %d", nameAndUrl.getUrl(), image.getTotal())); - log.debug(String.format("Starting data transfer for %s", nameAndUrl.getUrl())); - - byte[] buffer = new byte[READ_BUFFER_SIZE]; - int read; - while ((read = downloadStream.read(buffer, 0, READ_BUFFER_SIZE)) != -1 && !stopped) { - fos.write(buffer, 0, read); - image.increase(read); - downloadSpeedService.increase(read); - dataService.updateImageCurrent(image.getCurrent(), image.getId()); - } - fos.flush(); - EntityUtils.consumeQuietly(response.getEntity()); - if (stopped) { - return; - } - } - checkImageTypeAndRename( - dataService.findById(post.getId()).orElseThrow(), - outputFile, - nameAndUrl.getName(), - image.getIndex()); - } - } catch (Exception e) { - if (stopped) { - return; - } - throw new DownloadException(e); - } finally { - if (image.getCurrent() == image.getTotal() && image.getTotal() > 0) { - image.setStatus(Status.COMPLETE); - } else if (stopped) { - image.setStatus(Status.STOPPED); - } else { - image.setStatus(Status.ERROR); - } - dataService.updateImageStatus(image.getStatus(), image.getId()); - finished = true; - } - } - - private void checkImageTypeAndRename(Post post, File outputFile, String imageName, int index) - throws HostException { - try (ImageInputStream iis = ImageIO.createImageInputStream(outputFile)) { - Iterator it = ImageIO.getImageReaders(iis); - if (!it.hasNext()) { - throw new HostException("Image file is not recognized!"); - } - ImageReader reader = it.next(); - if (reader.getFormatName().equalsIgnoreCase("JPEG")) { - String imageNameLC = imageName.toLowerCase(); - if (!imageNameLC.endsWith("_jpg") && !imageNameLC.endsWith("_jpeg")) { - imageName += ".jpg"; - } else { - String toReplace = null; - if (imageNameLC.endsWith("_jpg")) { - toReplace = "_jpg"; - } else if (imageNameLC.endsWith("_jpeg")) { - toReplace = "_jpeg"; - } - if (toReplace != null) { - imageName = imageName.substring(0, imageName.length() - toReplace.length()) + ".jpg"; - } - } - } else if (reader.getFormatName().equalsIgnoreCase("PNG")) { - String imageNameLC = imageName.toLowerCase(); - if (!imageNameLC.endsWith("_png")) { - imageName += ".png"; - } else { - String toReplace = null; - if (imageNameLC.endsWith("_png")) { - toReplace = "_png"; - } - if (toReplace != null) { - imageName = imageName.substring(0, imageName.length() - toReplace.length()) + ".png"; - } - } - } - } catch (Exception e) { - throw new HostException("Failed to guess image format", e); - } - try { - pathService.getDirectoryAccess().lock(); - File downloadDestinationFolder = pathService.calcDownloadDirectory(post, settings); - File outImage = - new File( - downloadDestinationFolder, - (settings.getForceOrder() ? String.format("%03d_", index) : "") + imageName); - Files.copy(outputFile.toPath(), outImage.toPath(), StandardCopyOption.REPLACE_EXISTING); - } catch (Exception e) { - throw new HostException("Failed to rename the image", e); - } finally { - pathService.getDirectoryAccess().unlock(); - try { - Files.delete(outputFile.toPath()); - } catch (IOException e) { - log.warn(String.format("Failed to delete temporary file %s", outputFile.getAbsolutePath())); - } - } - } - - @Override - public void run() throws Exception { - if (stopped) { - finished = true; - return; - } - log.debug(String.format("Starting downloading %s", image.getUrl())); - download(post, image); - } - - @Override - public boolean equals(Object o) { - if (this == o) return true; - if (o == null || getClass() != o.getClass()) return false; - DownloadJob that = (DownloadJob) o; - return Objects.equals(image, that.image) && Objects.equals(post, that.post); - } - - @Override - public int hashCode() { - return Objects.hash(image, post); - } - - public void stop() { - this.stopped = true; - List requests = - (List) - this.context.getAttribute(ContextAttributes.OPEN_CONNECTION.toString()); - if (requests != null) { - for (AbstractExecutionAwareRequest request : requests) { - request.abort(); - } - } - } - - public enum ContextAttributes { - OPEN_CONNECTION("OPEN_CONNECTION"); - - private final String value; - - ContextAttributes(String value) { - this.value = value; - } - - @Override - public String toString() { - return value; - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/download/DownloadJobWrapper.java b/vripper-server/src/main/java/tn/mnlr/vripper/download/DownloadJobWrapper.java deleted file mode 100644 index 89e03cc9..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/download/DownloadJobWrapper.java +++ /dev/null @@ -1,70 +0,0 @@ -package tn.mnlr.vripper.download; - -import lombok.extern.slf4j.Slf4j; -import net.jodah.failsafe.Failsafe; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.Utils; -import tn.mnlr.vripper.jpa.domain.LogEvent; -import tn.mnlr.vripper.jpa.domain.enums.Status; -import tn.mnlr.vripper.jpa.repositories.ILogEventRepository; -import tn.mnlr.vripper.services.ConnectionService; -import tn.mnlr.vripper.services.DataService; - -import java.time.LocalDateTime; - -@Slf4j -public class DownloadJobWrapper implements Runnable { - - private final DownloadService downloadService; - private final DataService dataService; - private final ConnectionService connectionService; - private final ILogEventRepository eventRepository; - - private final DownloadJob downloadJob; - - public DownloadJobWrapper(final DownloadJob downloadJob) { - downloadService = SpringContext.getBean(DownloadService.class); - dataService = SpringContext.getBean(DataService.class); - connectionService = SpringContext.getBean(ConnectionService.class); - eventRepository = SpringContext.getBean(ILogEventRepository.class); - - this.downloadJob = downloadJob; - } - - @Override - public void run() { - Failsafe.with(connectionService.getRetryPolicy()) - .onFailure( - e -> { - try { - LogEvent logEvent = - new LogEvent( - LogEvent.Type.DOWNLOAD, - LogEvent.Status.ERROR, - LocalDateTime.now(), - String.format( - "Failed to download %s\n %s", - downloadJob.getImage().getUrl(), - Utils.throwableToString(e.getFailure()))); - eventRepository.save(logEvent); - } catch (Exception exp) { - log.error("Failed to save event", exp); - } - log.error( - String.format( - "Failed to download %s after %d tries", - downloadJob.getImage().getUrl(), e.getAttemptCount()), - e.getFailure()); - downloadJob.getImage().setStatus(Status.ERROR); - dataService.updateImageStatus( - downloadJob.getImage().getStatus(), downloadJob.getImage().getId()); - }) - .onComplete( - e -> { - dataService.afterJobFinish(downloadJob.getImage(), downloadJob.getPost()); - downloadService.afterJobFinish(downloadJob); - log.debug(String.format("Finished downloading %s", downloadJob.getImage().getUrl())); - }) - .run(downloadJob); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/download/DownloadService.java b/vripper-server/src/main/java/tn/mnlr/vripper/download/DownloadService.java deleted file mode 100644 index 4af11f76..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/download/DownloadService.java +++ /dev/null @@ -1,274 +0,0 @@ -package tn.mnlr.vripper.download; - -import lombok.NonNull; -import lombok.extern.slf4j.Slf4j; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.host.Host; -import tn.mnlr.vripper.jpa.domain.Image; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.jpa.domain.enums.Status; -import tn.mnlr.vripper.services.DataService; -import tn.mnlr.vripper.services.MetadataService; -import tn.mnlr.vripper.services.SettingsService; -import tn.mnlr.vripper.services.domain.Settings; - -import javax.annotation.PostConstruct; -import java.util.*; -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.TimeUnit; -import java.util.concurrent.atomic.AtomicInteger; -import java.util.stream.Collectors; - -@Service -@Slf4j -public class DownloadService { - - private final int MAX_POOL_SIZE = 12; - - private final SettingsService settingsService; - private final DataService dataService; - private final MetadataService metadataService; - private final List hosts; - - private final Map threadCount = new HashMap<>(); - private final ExecutorService executor = Executors.newFixedThreadPool(MAX_POOL_SIZE); - private final List running = new ArrayList<>(); - private final List pending = new ArrayList<>(); - - private final Thread pollThread; - - @Autowired - public DownloadService( - SettingsService settingsService, - DataService dataService, - MetadataService metadataService, - List hosts) { - this.settingsService = settingsService; - this.dataService = dataService; - this.metadataService = metadataService; - this.hosts = hosts; - pollThread = - new Thread( - () -> { - while (!Thread.interrupted()) { - try { - synchronized (this) { - List accepted = new ArrayList<>(); - List candidates = getCandidates(candidateCount()); - candidates.forEach( - c -> { - if (canRun(c.getImage().getHost())) { - accepted.add(c); - } - }); - pending.removeAll(accepted); - accepted.forEach(this::schedule); - accepted.clear(); - this.wait(); - } - } catch (InterruptedException e) { - Thread.currentThread().interrupt(); - break; - } - } - }, - "Download scheduler thread"); - } - - @PostConstruct - private void init() { - pollThread.start(); - } - - public void destroy() throws Exception { - log.info("Shutting down ExecutionService"); - pollThread.interrupt(); - executor.shutdown(); - dataService - .findAllPosts() - .forEach( - p -> { - log.debug(String.format("Stopping download jobs for %s", p)); - this.stopRunning(p.getPostId()); - }); - if (!executor.awaitTermination(5, TimeUnit.SECONDS)) { - log.warn("Some jobs are still running!, forcing shutdown"); - executor.shutdownNow(); - } - } - - private synchronized void stopRunning(@NonNull String postId) { - List stopping = new ArrayList<>(); - Iterator iterator = running.iterator(); - while (iterator.hasNext()) { - DownloadJob downloadJob = iterator.next(); - if (postId.equals(downloadJob.getPost().getPostId())) { - downloadJob.stop(); - iterator.remove(); - stopping.add(downloadJob); - } - } - - while (!stopping.isEmpty()) { - stopping.removeIf(DownloadJob::isFinished); - try { - Thread.sleep(100); - } catch (InterruptedException e) { - Thread.currentThread().interrupt(); - } - } - } - - public void stopAll(List postIds) { - stop( - Objects.requireNonNullElseGet( - postIds, - () -> - dataService.findAllPosts().stream() - .map(Post::getPostId) - .collect(Collectors.toList()))); - } - - public void restartAll(List posIds) { - restart( - Objects.requireNonNullElseGet( - posIds, - () -> - dataService.findAllPosts().stream() - .map(Post::getPostId) - .collect(Collectors.toList()))); - } - - private void restart(@NonNull List postIds) { - Map> data = new HashMap<>(); - for (String postId : postIds) { - if (isPending(postId)) { - log.warn( - String.format("Cannot restart, jobs are currently running for post id %s", postIds)); - continue; - } - List images = dataService.findByPostIdAndIsNotCompleted(postId); - if (images.isEmpty()) { - continue; - } - Post post = dataService.findPostByPostId(postId).orElseThrow(); - log.debug(String.format("Restarting %d jobs for post id %s", images.size(), postIds)); - data.put(post, images); - } - enqueue(data); - } - - private synchronized boolean isPending(String postId) { - return pending.stream().anyMatch(p -> p.getPost().getPostId().equals(postId)); - } - - private synchronized boolean isRunning(String postId) { - return running.stream().anyMatch(p -> p.getPost().getPostId().equals(postId)); - } - - private synchronized void stop(List postIds) { - - for (String postId : postIds) { - final Post post = dataService.findPostByPostId(postId).orElseThrow(); - if (post == null) { - continue; - } - pending.removeIf(p -> p.getPost().equals(post)); - stopRunning(postId); - dataService.stopImagesByPostIdAndIsNotCompleted(postId); - dataService.finishPost(post); - } - metadataService.stopFetchingMetadata(postIds); - } - - private boolean canRun(Host host) { - boolean canRun; - int totalRunning = threadCount.values().stream().mapToInt(AtomicInteger::get).sum(); - canRun = - threadCount.get(host).get() < settingsService.getSettings().getMaxThreads() - && (settingsService.getSettings().getMaxTotalThreads() == 0 - ? totalRunning < MAX_POOL_SIZE - : totalRunning < settingsService.getSettings().getMaxTotalThreads()); - if (canRun) { - threadCount.get(host).incrementAndGet(); - return true; - } - return false; - } - - private Map candidateCount() { - HashMap map = new HashMap<>(); - hosts.forEach( - h -> { - AtomicInteger count = threadCount.get(h); - if (count == null) { - count = new AtomicInteger(0); - threadCount.put(h, count); - } - map.put(h, settingsService.getSettings().getMaxThreads() - count.get()); - }); - return map; - } - - private List getCandidates(Map candidateCount) { - List candidates = new ArrayList<>(); - for (DownloadJob downloadJob : pending) { - Host host = downloadJob.getImage().getHost(); - Integer maxPerHost = candidateCount.get(host); - if (maxPerHost > 0) { - candidates.add(downloadJob); - candidateCount.put(host, maxPerHost - 1); - } - } - return candidates; - } - - public void enqueue(Map> images) { - synchronized (this) { - for (Map.Entry> entry : images.entrySet()) { - entry.getKey().setStatus(Status.PENDING); - dataService.updatePostStatus(entry.getKey().getStatus(), entry.getKey().getId()); - for (Image image : entry.getValue()) { - log.debug(String.format("Enqueuing a job for %s", image.getUrl())); - image.init(); - dataService.updateImageStatus(image.getStatus(), image.getId()); - dataService.updateImageCurrent(image.getCurrent(), image.getId()); - DownloadJob downloadJob = - new DownloadJob( - entry.getKey(), image, (Settings) settingsService.getSettings().clone()); - pending.add(downloadJob); - } - } - pending.sort(Comparator.comparing(e -> e.getPost().getAddedOn())); - this.notify(); - } - } - - private synchronized void schedule(DownloadJob downloadJob) { - log.debug(String.format("Scheduling a job for %s", downloadJob.getImage().getUrl())); - executor.execute(new DownloadJobWrapper(downloadJob)); - running.add(downloadJob); - } - - public void afterJobFinish(DownloadJob downloadJob) { - synchronized (this) { - running.remove(downloadJob); - threadCount.get(downloadJob.getImage().getHost()).decrementAndGet(); - if (!isPending(downloadJob.getPost().getPostId()) - && !isRunning(downloadJob.getPost().getPostId())) { - dataService.finishPost(downloadJob.getPost()); - } - this.notify(); - } - } - - public int pendingCount() { - return pending.size(); - } - - public int runningCount() { - return running.size(); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/event/Event.java b/vripper-server/src/main/java/tn/mnlr/vripper/event/Event.java deleted file mode 100644 index 985324d3..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/event/Event.java +++ /dev/null @@ -1,34 +0,0 @@ -package tn.mnlr.vripper.event; - -import lombok.Getter; - -@Getter -public class Event { - - private final T data; - private final Kind kind; - - private Event(Kind kind, T data) { - this.data = data; - this.kind = kind; - } - - public static Event wrap(Kind kind, T data) { - return new Event<>(kind, data); - } - - public enum Kind { - POST_UPDATE, - POST_REMOVE, - IMAGE_UPDATE, - METADATA_UPDATE, - QUEUED_UPDATE, - QUEUED_REMOVE, - LOG_EVENT_UPDATE, - LOG_EVENT_REMOVE, - VG_USER, - GLOBAL_STATE, - BYTES_PER_SECOND, - SETTINGS_UPDATE - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/event/EventBus.java b/vripper-server/src/main/java/tn/mnlr/vripper/event/EventBus.java deleted file mode 100644 index 0f1634dd..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/event/EventBus.java +++ /dev/null @@ -1,27 +0,0 @@ -package tn.mnlr.vripper.event; - -import org.springframework.stereotype.Service; -import reactor.core.publisher.Flux; -import reactor.core.publisher.Sinks; - -import javax.annotation.PreDestroy; - -@Service -public class EventBus { - - public static final Sinks.EmitFailureHandler RETRY = (signalType, emitResult) -> true; - private final Sinks.Many> sink = Sinks.many().multicast().onBackpressureBuffer(); - - public void publishEvent(Event event) { - sink.emitNext(event, RETRY); - } - - public Flux> flux() { - return sink.asFlux(); - } - - @PreDestroy - private void destroy() { - sink.emitComplete(RETRY); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/exception/DownloadException.java b/vripper-server/src/main/java/tn/mnlr/vripper/exception/DownloadException.java deleted file mode 100644 index 49c65246..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/exception/DownloadException.java +++ /dev/null @@ -1,12 +0,0 @@ -package tn.mnlr.vripper.exception; - -public class DownloadException extends Exception { - - public DownloadException(String message) { - super(message); - } - - public DownloadException(Throwable e) { - super(e); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/exception/HostException.java b/vripper-server/src/main/java/tn/mnlr/vripper/exception/HostException.java deleted file mode 100644 index 0a53a8a8..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/exception/HostException.java +++ /dev/null @@ -1,15 +0,0 @@ -package tn.mnlr.vripper.exception; - -public class HostException extends Exception { - public HostException(Throwable e) { - super(e); - } - - public HostException(String message) { - super(message); - } - - public HostException(String message, Throwable e) { - super(message, e); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/exception/HtmlProcessorException.java b/vripper-server/src/main/java/tn/mnlr/vripper/exception/HtmlProcessorException.java deleted file mode 100644 index 61bebe07..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/exception/HtmlProcessorException.java +++ /dev/null @@ -1,8 +0,0 @@ -package tn.mnlr.vripper.exception; - -public class HtmlProcessorException extends Exception { - - public HtmlProcessorException(Throwable e) { - super(e); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/exception/PostParseException.java b/vripper-server/src/main/java/tn/mnlr/vripper/exception/PostParseException.java deleted file mode 100644 index dcd6bede..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/exception/PostParseException.java +++ /dev/null @@ -1,16 +0,0 @@ -package tn.mnlr.vripper.exception; - -public class PostParseException extends Exception { - - public PostParseException(String message) { - super(message); - } - - public PostParseException(String message, Throwable e) { - super(message, e); - } - - public PostParseException(Throwable e) { - super(e); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/exception/QueueException.java b/vripper-server/src/main/java/tn/mnlr/vripper/exception/QueueException.java deleted file mode 100644 index 7d545bdb..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/exception/QueueException.java +++ /dev/null @@ -1,16 +0,0 @@ -package tn.mnlr.vripper.exception; - -public class QueueException extends Exception { - - public QueueException(String message) { - super(message); - } - - public QueueException(String message, Throwable e) { - super(message, e); - } - - public QueueException(Throwable e) { - super(e); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/exception/RenameException.java b/vripper-server/src/main/java/tn/mnlr/vripper/exception/RenameException.java deleted file mode 100644 index 3d8a2687..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/exception/RenameException.java +++ /dev/null @@ -1,10 +0,0 @@ -package tn.mnlr.vripper.exception; - -import java.io.IOException; - -public class RenameException extends Exception { - - public RenameException(String message, IOException e) { - super(message, e); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/exception/ValidationException.java b/vripper-server/src/main/java/tn/mnlr/vripper/exception/ValidationException.java deleted file mode 100644 index 5d3d3ae5..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/exception/ValidationException.java +++ /dev/null @@ -1,8 +0,0 @@ -package tn.mnlr.vripper.exception; - -public class ValidationException extends Exception { - - public ValidationException(String message) { - super(message); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/exception/VripperException.java b/vripper-server/src/main/java/tn/mnlr/vripper/exception/VripperException.java deleted file mode 100644 index 74a14e9c..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/exception/VripperException.java +++ /dev/null @@ -1,11 +0,0 @@ -package tn.mnlr.vripper.exception; - -public class VripperException extends Exception { - public VripperException(String message) { - super(message); - } - - public VripperException(Throwable e) { - super(e); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/exception/XpathException.java b/vripper-server/src/main/java/tn/mnlr/vripper/exception/XpathException.java deleted file mode 100644 index cf359d77..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/exception/XpathException.java +++ /dev/null @@ -1,7 +0,0 @@ -package tn.mnlr.vripper.exception; - -public class XpathException extends Exception { - public XpathException(Throwable e) { - super(e); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/AcidimgHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/AcidimgHost.java deleted file mode 100644 index a8bf628b..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/AcidimgHost.java +++ /dev/null @@ -1,122 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.NameValuePair; -import org.apache.http.client.HttpClient; -import org.apache.http.client.entity.UrlEncodedFormEntity; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.client.protocol.HttpClientContext; -import org.apache.http.message.BasicNameValuePair; -import org.apache.http.util.EntityUtils; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.ConnectionService; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.HtmlProcessorService; -import tn.mnlr.vripper.services.XpathService; - -import java.util.ArrayList; -import java.util.List; - -@Service -@Slf4j -public class AcidimgHost extends Host { - - private static final String host = "acidimg.cc"; - private static final String CONTINUE_BUTTON_XPATH = "//input[@id='continuebutton']"; - private static final String IMG_XPATH = "//img[@class='centred']"; - - private final ConnectionService cm; - private final HostService hostService; - private final XpathService xpathService; - private final HtmlProcessorService htmlProcessorService; - - @Autowired - public AcidimgHost( - ConnectionService cm, - HostService hostService, - XpathService xpathService, - HtmlProcessorService htmlProcessorService) { - this.cm = cm; - this.hostService = hostService; - this.xpathService = xpathService; - this.htmlProcessorService = htmlProcessorService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String url, final HttpClientContext context) - throws HostException { - - Document doc = hostService.getResponse(url, context).getDocument(); - - Node contDiv; - try { - log.debug(String.format("Looking for xpath expression %s in %s", CONTINUE_BUTTON_XPATH, url)); - contDiv = xpathService.getAsNode(doc, CONTINUE_BUTTON_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - if (contDiv != null) { - log.debug(String.format("Click button found for %s", url)); - HttpClient client = cm.getClient().build(); - HttpPost httpPost = cm.buildHttpPost(url, context); - httpPost.addHeader("Referer", url); - List params = new ArrayList<>(); - params.add(new BasicNameValuePair("imgContinue", "Continue to your image")); - try { - httpPost.setEntity(new UrlEncodedFormEntity(params)); - } catch (Exception e) { - throw new HostException(e); - } - - log.debug(String.format("Requesting %s", httpPost)); - try (CloseableHttpResponse response = - (CloseableHttpResponse) client.execute(httpPost, context)) { - log.debug(String.format("Cleaning response for %s", httpPost)); - doc = htmlProcessorService.clean(EntityUtils.toString(response.getEntity())); - EntityUtils.consumeQuietly(response.getEntity()); - } catch (Exception e) { - throw new HostException(e); - } - } - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - if (imgNode == null) { - throw new HostException("Cannot find the image node"); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = imgNode.getAttributes().getNamedItem("alt").getTextContent().trim(); - String imgUrl = imgNode.getAttributes().getNamedItem("src").getTextContent().trim(); - - return new HostService.NameUrl( - imgTitle.isEmpty() ? hostService.getDefaultImageName(imgUrl) : imgTitle, imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/DPicMeHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/DPicMeHost.java deleted file mode 100644 index 40e03b28..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/DPicMeHost.java +++ /dev/null @@ -1,85 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -import java.util.Optional; -import java.util.UUID; - -@Service -@Slf4j -public class DPicMeHost extends Host { - - private static final String IMG_XPATH = "//img[@id='pic']"; - private static final String host = "dpic.me"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public DPicMeHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String url, final HttpClientContext context) - throws HostException { - - HostService.Response response = - hostService.getResponse(url.replace("http://", "https://"), context); - Document doc = response.getDocument(); - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - if (imgNode == null) { - throw new HostException(String.format("Xpath '%s' cannot be found in '%s'", IMG_XPATH, url)); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = - Optional.ofNullable(imgNode.getAttributes().getNamedItem("alt")) - .map(e -> e.getTextContent().trim()) - .orElse(""); - String imgUrl = - Optional.ofNullable(imgNode.getAttributes().getNamedItem("src")) - .map(e -> e.getTextContent().trim()) - .orElse(""); - String defaultName = UUID.randomUUID().toString(); - - int index = imgUrl.lastIndexOf('/'); - if (index != -1 && index < imgUrl.length()) { - defaultName = imgUrl.substring(imgUrl.lastIndexOf('/') + 1); - } - - return new HostService.NameUrl(imgTitle.isEmpty() ? defaultName : imgTitle, imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/Host.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/Host.java deleted file mode 100644 index d7d3c38d..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/Host.java +++ /dev/null @@ -1,43 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.services.HostService; - -import java.util.Objects; - -@Service -@Slf4j -public abstract class Host { - - public abstract String getHost(); - - public abstract String getLookup(); - - public boolean isSupported(String url) { - return url.contains(getLookup()); - } - - public abstract HostService.NameUrl getNameAndUrl( - final String url, final HttpClientContext context) throws HostException; - - @Override - public boolean equals(Object o) { - if (this == o) return true; - if (o == null || getClass() != o.getClass()) return false; - Host host = (Host) o; - return Objects.equals(getHost(), host.getHost()); - } - - @Override - public int hashCode() { - return Objects.hash(getHost()); - } - - @Override - public String toString() { - return getHost(); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageBamHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageBamHost.java deleted file mode 100644 index 04741eb3..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageBamHost.java +++ /dev/null @@ -1,96 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -import java.util.Optional; -import java.util.UUID; - -@Service -@Slf4j -public class ImageBamHost extends Host { - - private static final String host = "imagebam.com"; - private static final String IMG_XPATH = "//img[contains(@class,'main-image')]"; - private static final String CONTINUE_XPATH = "//*[contains(text(), 'Continue')]"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public ImageBamHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String url, final HttpClientContext context) - throws HostException { - - HostService.Response response = hostService.getResponse(url, context); - Document doc = response.getDocument(); - - try { - log.debug(String.format("Looking for xpath expression %s in %s", CONTINUE_XPATH, url)); - if (xpathService.getAsNode(doc, CONTINUE_XPATH) != null) { - // Button detected. No need to actually click it, just make the call again. - response = hostService.getResponse(url, context); - doc = response.getDocument(); - } - } catch (XpathException e) { - throw new HostException(e); - } - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - if (imgNode == null) { - throw new HostException(String.format("Xpath '%s' cannot be found in '%s'", IMG_XPATH, url)); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = - Optional.ofNullable(imgNode.getAttributes().getNamedItem("alt")) - .map(e -> e.getTextContent().trim()) - .orElse(""); - String imgUrl = - Optional.ofNullable(imgNode.getAttributes().getNamedItem("src")) - .map(e -> e.getTextContent().trim()) - .orElse(""); - String defaultName = UUID.randomUUID().toString(); - - int index = imgUrl.lastIndexOf('/'); - if (index != -1 && index < imgUrl.length()) { - defaultName = imgUrl.substring(imgUrl.lastIndexOf('/') + 1); - } - - return new HostService.NameUrl(imgTitle.isEmpty() ? defaultName : imgTitle, imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageTwistHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageTwistHost.java deleted file mode 100644 index ad8fb73f..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageTwistHost.java +++ /dev/null @@ -1,70 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -import java.util.Optional; - -@Service -@Slf4j -public class ImageTwistHost extends Host { - - private static final String IMG_XPATH = "//img[contains(@class, 'img')]"; - private static final String host = "imagetwist.com"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public ImageTwistHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String url, final HttpClientContext context) - throws HostException { - - Document doc = hostService.getResponse(url, context).getDocument(); - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = - Optional.ofNullable(imgNode.getAttributes().getNamedItem("alt")) - .map(Node::getTextContent) - .map(String::trim) - .orElse(null); - String imgUrl = imgNode.getAttributes().getNamedItem("src").getTextContent().trim(); - - return new HostService.NameUrl(imgTitle, imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageVenueHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageVenueHost.java deleted file mode 100644 index 9b6a8336..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageVenueHost.java +++ /dev/null @@ -1,84 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -@Service -@Slf4j -public class ImageVenueHost extends Host { - - private static final String host = "imagevenue.com"; - private static final String CONTINUE_BUTTON_XPATH = "//a[@title='Continue to ImageVenue']"; - private static final String IMG_XPATH = "//a[@data-toggle='full']/img"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public ImageVenueHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String _url, final HttpClientContext context) - throws HostException { - - String url = _url.replace("http://", "https://"); - - HostService.Response resp = hostService.getResponse(url, context); - Document doc = resp.getDocument(); - - try { - log.debug(String.format("Looking for xpath expression %s in %s", CONTINUE_BUTTON_XPATH, url)); - if (xpathService.getAsNode(doc, CONTINUE_BUTTON_XPATH) != null) { - // Button detected. No need to actually click it, just make the call again. - resp = hostService.getResponse(url, context); - doc = resp.getDocument(); - } - } catch (XpathException e) { - throw new HostException(e); - } - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - if (imgNode == null) { - throw new HostException("Failed to locate image"); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = imgNode.getAttributes().getNamedItem("alt").getTextContent().trim(); - String imgUrl = imgNode.getAttributes().getNamedItem("src").getTextContent().trim(); - - return new HostService.NameUrl( - imgTitle.isEmpty() ? imgUrl.substring(imgUrl.lastIndexOf('/') + 1) : imgTitle, imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageZillaHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageZillaHost.java deleted file mode 100644 index fe166576..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImageZillaHost.java +++ /dev/null @@ -1,71 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -@Service -@Slf4j -public class ImageZillaHost extends Host { - - private static final String host = "imagezilla.net"; - private static final String lookup = "imagezilla.net/show"; - private static final String IMG_XPATH = "//img[@id='photo']"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public ImageZillaHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return lookup; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String url, final HttpClientContext context) - throws HostException { - - Document doc = hostService.getResponse(url, context).getDocument(); - - String title; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - Node titleNode = xpathService.getAsNode(doc, IMG_XPATH).getAttributes().getNamedItem("title"); - log.debug(String.format("Resolving name for %s", url)); - if (titleNode != null) { - title = titleNode.getTextContent().trim(); - } else { - title = null; - } - } catch (XpathException e) { - throw new HostException(e); - } - - if (title == null || title.isEmpty()) { - title = hostService.getDefaultImageName(url); - } - - try { - return new HostService.NameUrl(title, url.replace("show", "images")); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImgSpiceHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/ImgSpiceHost.java deleted file mode 100644 index 205bb996..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImgSpiceHost.java +++ /dev/null @@ -1,67 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -@Service -@Slf4j -public class ImgSpiceHost extends Host { - - private static final String host = "imgspice.com"; - private static final String IMG_XPATH = "//img[@id='imgpreview']"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public ImgSpiceHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String _url, final HttpClientContext context) - throws HostException { - - String url = _url.replace("http://", "https://"); - HostService.Response resp = hostService.getResponse(url, context); - Document doc = resp.getDocument(); - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = imgNode.getAttributes().getNamedItem("alt").getTextContent().trim(); - String imgUrl = imgNode.getAttributes().getNamedItem("src").getTextContent().trim(); - - return new HostService.NameUrl( - imgTitle.isEmpty() ? imgUrl.substring(imgUrl.lastIndexOf('/') + 1) : imgTitle, imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImgboxHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/ImgboxHost.java deleted file mode 100644 index 593facf1..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImgboxHost.java +++ /dev/null @@ -1,64 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -@Service -@Slf4j -public class ImgboxHost extends Host { - - private static final String host = "imgbox.com"; - private static final String IMG_XPATH = "//img[@id='img']"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public ImgboxHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String url, final HttpClientContext context) - throws HostException { - - Document doc = hostService.getResponse(url, context).getDocument(); - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = imgNode.getAttributes().getNamedItem("title").getTextContent().trim(); - String imgUrl = imgNode.getAttributes().getNamedItem("src").getTextContent().trim(); - - return new HostService.NameUrl(imgTitle, imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImxHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/ImxHost.java deleted file mode 100644 index b4167023..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/ImxHost.java +++ /dev/null @@ -1,129 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.NameValuePair; -import org.apache.http.client.HttpClient; -import org.apache.http.client.entity.UrlEncodedFormEntity; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.client.protocol.HttpClientContext; -import org.apache.http.message.BasicNameValuePair; -import org.apache.http.util.EntityUtils; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.HtmlProcessorException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.ConnectionService; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.HtmlProcessorService; -import tn.mnlr.vripper.services.XpathService; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.List; - -@Service -@Slf4j -public class ImxHost extends Host { - - private static final String host = "imx.to"; - private static final String CONTINUE_BUTTON_XPATH = "//*[@name='imgContinue']"; - private static final String IMG_XPATH = "//img[@class='centred']"; - - private final HostService hostService; - private final XpathService xpathService; - private final ConnectionService cm; - private final HtmlProcessorService htmlProcessorService; - - @Autowired - public ImxHost( - HostService hostService, - XpathService xpathService, - ConnectionService cm, - HtmlProcessorService htmlProcessorService) { - this.hostService = hostService; - this.xpathService = xpathService; - this.cm = cm; - this.htmlProcessorService = htmlProcessorService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String _url, final HttpClientContext context) - throws HostException { - - String url = _url.replace("http://", "https://"); - HostService.Response resp = hostService.getResponse(url, context); - Document doc = resp.getDocument(); - - Node contDiv; - String value = null; - try { - log.debug(String.format("Looking for xpath expression %s in %s", CONTINUE_BUTTON_XPATH, url)); - contDiv = xpathService.getAsNode(doc, CONTINUE_BUTTON_XPATH); - if (contDiv == null) { - throw new HostException(CONTINUE_BUTTON_XPATH + " cannot be found"); - } - Node node = contDiv.getAttributes().getNamedItem("value"); - if (node != null) { - value = node.getTextContent(); - } - } catch (XpathException e) { - throw new HostException(e); - } - - if (value == null) { - throw new HostException("Failed to obtain value attribute from continue input"); - } - log.debug(String.format("Click button found for %s", url)); - HttpClient client = cm.getClient().build(); - HttpPost httpPost = cm.buildHttpPost(url, context); - List params = new ArrayList<>(); - params.add(new BasicNameValuePair("imgContinue", value)); - try { - httpPost.setEntity(new UrlEncodedFormEntity(params)); - } catch (Exception e) { - throw new HostException(e); - } - log.debug(String.format("Requesting %s", httpPost)); - try (CloseableHttpResponse response = - (CloseableHttpResponse) client.execute(httpPost, context)) { - log.debug(String.format("Cleaning response for %s", httpPost)); - doc = htmlProcessorService.clean(EntityUtils.toString(response.getEntity())); - EntityUtils.consumeQuietly(response.getEntity()); - } catch (IOException | HtmlProcessorException e) { - throw new HostException(e); - } - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = imgNode.getAttributes().getNamedItem("alt").getTextContent().trim(); - String imgUrl = imgNode.getAttributes().getNamedItem("src").getTextContent().trim(); - - return new HostService.NameUrl( - imgTitle.isEmpty() ? imgUrl.substring(imgUrl.lastIndexOf('/') + 1) : imgTitle, imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/PimpandhostHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/PimpandhostHost.java deleted file mode 100644 index cc79b460..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/PimpandhostHost.java +++ /dev/null @@ -1,75 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -@Service -@Slf4j -public class PimpandhostHost extends Host { - - private static final String host = "pimpandhost.com"; - private static final String IMG_XPATH = "//img[contains(@class, 'original')]"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public PimpandhostHost(HostService hostService, XpathService xpathService) { - super(); - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String _url, final HttpClientContext context) - throws HostException { - - String url; - try { - url = hostService.appendUri(_url.replace("http://", "https://"), "size=original"); - } catch (Exception e) { - throw new HostException(e); - } - - HostService.Response resp = hostService.getResponse(url, context); - Document doc = resp.getDocument(); - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = imgNode.getAttributes().getNamedItem("alt").getTextContent().trim(); - String imgUrl = - "https:" + imgNode.getAttributes().getNamedItem("src").getTextContent().trim(); - - return new HostService.NameUrl( - imgTitle.isEmpty() ? imgUrl.substring(imgUrl.lastIndexOf('/') + 1) : imgTitle, imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/PixRouteHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/PixRouteHost.java deleted file mode 100644 index 767a6347..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/PixRouteHost.java +++ /dev/null @@ -1,69 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -@Service -@Slf4j -public class PixRouteHost extends Host { - - private static final String host = "pixroute.com"; - private static final String IMG_XPATH = "//img[@id='imgpreview']"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public PixRouteHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String _url, final HttpClientContext context) - throws HostException { - - String url = _url.replace("http://", "https://"); - Document doc = hostService.getResponse(url, context).getDocument(); - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - if (imgNode == null) { - throw new HostException("Failed to locate image"); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - - return new HostService.NameUrl( - imgNode.getAttributes().getNamedItem("alt").getTextContent().trim(), - imgNode.getAttributes().getNamedItem("src").getTextContent().trim()); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/PixhostHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/PixhostHost.java deleted file mode 100644 index c61cad8c..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/PixhostHost.java +++ /dev/null @@ -1,65 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -@Service -@Slf4j -public class PixhostHost extends Host { - - private static final String host = "pixhost.to"; - private static final String lookup = "pixhost.to/show"; - private static final String IMG_XPATH = "//img[@id='image']"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public PixhostHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return lookup; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String url, final HttpClientContext context) - throws HostException { - - Document doc = hostService.getResponse(url, context).getDocument(); - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = imgNode.getAttributes().getNamedItem("alt").getTextContent().trim(); - String imgUrl = imgNode.getAttributes().getNamedItem("src").getTextContent().trim(); - - return new HostService.NameUrl(imgTitle.substring(imgTitle.indexOf('_') + 1), imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/PixxxelsHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/PixxxelsHost.java deleted file mode 100644 index b0d899ae..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/PixxxelsHost.java +++ /dev/null @@ -1,72 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -@Service -@Slf4j -public class PixxxelsHost extends Host { - - private static final String host = "pixxxels.cc"; - private static final String IMG_XPATH = "//*[@id='download']"; - private static final String TITLE_XPATH = "//*[contains(@class,'imagename')]"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public PixxxelsHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String _url, final HttpClientContext context) - throws HostException { - - String url = _url.replace("http://", "https://"); - HostService.Response resp = hostService.getResponse(url, context); - Document doc = resp.getDocument(); - - Node imgNode, titleNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - - log.debug(String.format("Looking for xpath expression %s in %s", TITLE_XPATH, url)); - titleNode = xpathService.getAsNode(doc, TITLE_XPATH); - - } catch (XpathException e) { - throw new HostException(e); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = titleNode.getTextContent().trim(); - String imgUrl = imgNode.getAttributes().getNamedItem("href").getTextContent().trim(); - - return new HostService.NameUrl( - imgTitle.isEmpty() ? imgUrl.substring(imgUrl.lastIndexOf('/') + 1) : imgTitle, imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/PostImgHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/PostImgHost.java deleted file mode 100644 index 6456bbf3..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/PostImgHost.java +++ /dev/null @@ -1,74 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -import java.util.Optional; - -@Service -@Slf4j -public class PostImgHost extends Host { - - private static final String host = "postimg.cc"; - private static final String TITLE_XPATH = "//span[contains(@class,'imagename')]"; - private static final String IMG_XPATH = "//a[@id='download']"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public PostImgHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String _url, final HttpClientContext context) - throws HostException { - - String url = _url.replace("http://", "https://"); - Document doc = hostService.getResponse(url, context).getDocument(); - - Node urlNode, titleNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", TITLE_XPATH, url)); - titleNode = xpathService.getAsNode(doc, TITLE_XPATH); - - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - urlNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = - Optional.ofNullable(titleNode) - .map(node -> node.getTextContent().trim()) - .orElseGet(() -> hostService.getDefaultImageName(url)); - - return new HostService.NameUrl( - imgTitle, urlNode.getAttributes().getNamedItem("href").getTextContent().trim()); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/TurboImageHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/TurboImageHost.java deleted file mode 100644 index 027a7cbd..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/TurboImageHost.java +++ /dev/null @@ -1,73 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -@Service -@Slf4j -public class TurboImageHost extends Host { - - private static final String host = "turboimagehost.com"; - private static final String TITLE_XPATH = "//div[contains(@class,'titleFullS')]/h1"; - private static final String IMG_XPATH = "//img[@id='imageid']"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public TurboImageHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String url, final HttpClientContext context) - throws HostException { - - Document doc = hostService.getResponse(url, context).getDocument(); - - String title; - try { - log.debug(String.format("Looking for xpath expression %s in %s", TITLE_XPATH, url)); - Node titleNode = xpathService.getAsNode(doc, TITLE_XPATH); - log.debug(String.format("Resolving name for %s", url)); - if (titleNode != null) { - title = titleNode.getTextContent().trim(); - } else { - title = null; - } - } catch (XpathException e) { - throw new HostException(e); - } - - if (title == null || title.isEmpty()) { - title = hostService.getDefaultImageName(url); - } - - try { - Node urlNode = xpathService.getAsNode(doc, IMG_XPATH); - return new HostService.NameUrl( - title, urlNode.getAttributes().getNamedItem("src").getTextContent().trim()); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/host/ViprImHost.java b/vripper-server/src/main/java/tn/mnlr/vripper/host/ViprImHost.java deleted file mode 100644 index 6aa9e2d0..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/host/ViprImHost.java +++ /dev/null @@ -1,70 +0,0 @@ -package tn.mnlr.vripper.host; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.client.protocol.HttpClientContext; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.XpathException; -import tn.mnlr.vripper.services.HostService; -import tn.mnlr.vripper.services.XpathService; - -import java.util.Optional; - -@Service -@Slf4j -public class ViprImHost extends Host { - - private static final String IMG_XPATH = "//img[contains(@class, 'img')]"; - private static final String host = "vipr.im"; - - private final HostService hostService; - private final XpathService xpathService; - - @Autowired - public ViprImHost(HostService hostService, XpathService xpathService) { - this.hostService = hostService; - this.xpathService = xpathService; - } - - @Override - public String getHost() { - return host; - } - - @Override - public String getLookup() { - return host; - } - - @Override - public HostService.NameUrl getNameAndUrl(final String url, final HttpClientContext context) - throws HostException { - - Document doc = hostService.getResponse(url, context).getDocument(); - - Node imgNode; - try { - log.debug(String.format("Looking for xpath expression %s in %s", IMG_XPATH, url)); - imgNode = xpathService.getAsNode(doc, IMG_XPATH); - } catch (XpathException e) { - throw new HostException(e); - } - - try { - log.debug(String.format("Resolving name and image url for %s", url)); - String imgTitle = - Optional.ofNullable(imgNode.getAttributes().getNamedItem("alt")) - .map(Node::getTextContent) - .map(String::trim) - .orElse(null); - String imgUrl = imgNode.getAttributes().getNamedItem("src").getTextContent().trim(); - - return new HostService.NameUrl(imgTitle, imgUrl); - } catch (Exception e) { - throw new HostException("Unexpected error occurred", e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/Management.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/Management.java deleted file mode 100644 index dbe10a4c..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/Management.java +++ /dev/null @@ -1,79 +0,0 @@ -package tn.mnlr.vripper.jpa; - -import org.springframework.beans.factory.annotation.Value; -import org.springframework.jdbc.core.JdbcTemplate; -import org.springframework.scheduling.annotation.EnableScheduling; -import org.springframework.scheduling.annotation.Scheduled; -import org.springframework.stereotype.Component; -import tn.mnlr.vripper.download.DownloadService; -import tn.mnlr.vripper.services.ThreadPoolService; - -import javax.annotation.PostConstruct; -import javax.annotation.PreDestroy; -import java.io.File; -import java.time.LocalDate; -import java.time.LocalDateTime; -import java.time.format.DateTimeFormatter; -import java.util.Optional; -import java.util.regex.Matcher; -import java.util.regex.Pattern; - -@Component -@EnableScheduling -public class Management { - - public static final DateTimeFormatter FORMATTER = DateTimeFormatter.ofPattern("yyyy-MM-dd"); - private static final Pattern BACKUP_FILE_PATTERN = - Pattern.compile("^db_(\\d{4}-\\d{2}-\\d{2})\\.tar\\.gz$"); - - private final JdbcTemplate jdbcTemplate; - private final String backupFolder; - private final ThreadPoolService threadPoolService; - private final DownloadService downloadService; - - public Management( - JdbcTemplate jdbcTemplate, - @Value("${base.dir}") String baseDir, - @Value("${base.dir.name}") String baseDirName, - ThreadPoolService threadPoolService, - DownloadService downloadService) { - this.jdbcTemplate = jdbcTemplate; - this.threadPoolService = threadPoolService; - this.downloadService = downloadService; - this.backupFolder = baseDir + File.separator + baseDirName + File.separator + "backup"; - } - - @PreDestroy - public void destroy() { - - try { - threadPoolService.destroy(); - downloadService.destroy(); - this.jdbcTemplate.execute("SHUTDOWN"); - } catch (Exception ignored) { - Thread.currentThread().interrupt(); - } - } - - @PostConstruct - @Scheduled(cron = "0 0 0 ? * *") - private void backup() { - - for (File file : - Optional.ofNullable(new File(backupFolder).listFiles()).orElse(new File[] {})) { - Matcher matcher = BACKUP_FILE_PATTERN.matcher(file.getName()); - if (matcher.find()) { - LocalDate localDate = LocalDate.parse(matcher.group(1), FORMATTER); - if (localDate.isEqual(LocalDate.now())) { - return; - } - } - } - - // create a backup file - File backupFile = - new File( - backupFolder, String.format("db_%s.tar.gz", LocalDateTime.now().format(FORMATTER))); - jdbcTemplate.execute(String.format("BACKUP DATABASE TO '%s' BLOCKING", backupFile.getPath())); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/DateTimeSerializer.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/DateTimeSerializer.java deleted file mode 100644 index a33ba1fe..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/DateTimeSerializer.java +++ /dev/null @@ -1,25 +0,0 @@ -package tn.mnlr.vripper.jpa.domain; - -import com.fasterxml.jackson.core.JsonGenerator; -import com.fasterxml.jackson.databind.SerializerProvider; -import com.fasterxml.jackson.databind.ser.std.StdSerializer; - -import java.io.IOException; -import java.time.LocalDateTime; -import java.time.format.DateTimeFormatter; - -public class DateTimeSerializer extends StdSerializer { - - private static final DateTimeFormatter DATE_TIME_FORMATTER = - DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss.SSS"); - - protected DateTimeSerializer() { - super(LocalDateTime.class); - } - - @Override - public void serialize(LocalDateTime value, JsonGenerator gen, SerializerProvider provider) - throws IOException { - gen.writeString(value.format(DATE_TIME_FORMATTER)); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Image.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Image.java deleted file mode 100644 index 07a0b6a4..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Image.java +++ /dev/null @@ -1,66 +0,0 @@ -package tn.mnlr.vripper.jpa.domain; - -import com.fasterxml.jackson.annotation.JsonIgnore; -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; -import tn.mnlr.vripper.host.Host; -import tn.mnlr.vripper.jpa.domain.enums.Status; - -import java.util.Objects; - -@Getter -@Setter -@ToString -@NoArgsConstructor -public class Image { - - @JsonIgnore protected Long id; - - @JsonIgnore private Host host; - - private String url; - - private int index; - - private long current = 0; - - private long total = 0; - - private Status status; - - private String postId; - - @JsonIgnore private Long postIdRef; - - public Image(String postId, String url, Host host, int index) { - this.postId = postId; - this.url = url; - this.host = host; - this.index = index; - status = Status.STOPPED; - } - - public void increase(int read) { - current += read; - } - - public void init() { - current = 0; - status = Status.STOPPED; - } - - @Override - public boolean equals(Object o) { - if (this == o) return true; - if (o == null || getClass() != o.getClass()) return false; - Image image = (Image) o; - return Objects.equals(url, image.url); - } - - @Override - public int hashCode() { - return Objects.hash(url); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/LogEvent.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/LogEvent.java deleted file mode 100644 index 74ea3392..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/LogEvent.java +++ /dev/null @@ -1,52 +0,0 @@ -package tn.mnlr.vripper.jpa.domain; - -import com.fasterxml.jackson.databind.annotation.JsonSerialize; -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; - -import java.time.LocalDateTime; - -@Getter -@Setter -@ToString -@NoArgsConstructor -public class LogEvent { - - private Long id; - private Type type; - private Status status; - - @JsonSerialize(using = DateTimeSerializer.class) - private LocalDateTime time; - - private String message; - - public LogEvent(Type type, Status status, LocalDateTime time, String message) { - this.type = type; - this.status = status; - this.time = time; - this.message = message; - } - - public enum Type { - POST, - QUEUED, - THANKS, - METADATA, - SCAN, - DOWNLOAD, - - // Below are deprecated events - METADATA_CACHE_MISS, - QUEUED_CACHE_MISS - } - - public enum Status { - PENDING, - PROCESSING, - DONE, - ERROR - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Metadata.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Metadata.java deleted file mode 100644 index adaf61d2..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Metadata.java +++ /dev/null @@ -1,31 +0,0 @@ -package tn.mnlr.vripper.jpa.domain; - -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; - -import java.util.Collections; -import java.util.List; - -@Getter -@Setter -@NoArgsConstructor -public class Metadata { - - private Long postIdRef; - - private String postId; - - private String postedBy; - - private List resolvedNames = Collections.emptyList(); - - public static Metadata from(Metadata metadata) { - Metadata copy = new Metadata(); - copy.postIdRef = metadata.postIdRef; - copy.postId = metadata.postId; - copy.postedBy = metadata.postedBy; - copy.resolvedNames = List.copyOf(metadata.resolvedNames); - return copy; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Post.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Post.java deleted file mode 100644 index 76f110da..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Post.java +++ /dev/null @@ -1,90 +0,0 @@ -package tn.mnlr.vripper.jpa.domain; - -import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.databind.annotation.JsonSerialize; -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; -import tn.mnlr.vripper.jpa.domain.enums.Status; - -import java.time.LocalDateTime; -import java.util.Collections; -import java.util.Objects; -import java.util.Set; - -@Getter -@Setter -@ToString -@NoArgsConstructor -public class Post { - - @JsonIgnore private Long id; - - private Status status; - - private String postId; - - private String threadTitle; - - private String threadId; - - private String title; - - private String url; - - private int done; - - private int total; - - private Set hosts; - - private String forum; - - @JsonIgnore private String securityToken; - - @JsonIgnore private String downloadDirectory; - - private boolean thanked; - - private Set previews = Collections.emptySet(); - - private Metadata metadata; - - private int rank = Integer.MAX_VALUE; - - @JsonSerialize(using = DateTimeSerializer.class) - private LocalDateTime addedOn; - - public Post( - String title, - String url, - String postId, - String threadId, - String threadTitle, - String forum, - String securityToken) { - this.title = title; - this.url = url; - this.postId = postId; - this.forum = forum; - this.threadId = threadId; - this.threadTitle = threadTitle; - this.securityToken = securityToken; - status = Status.STOPPED; - addedOn = LocalDateTime.now(); - } - - @Override - public boolean equals(Object o) { - if (this == o) return true; - if (o == null || getClass() != o.getClass()) return false; - Post post = (Post) o; - return Objects.equals(postId, post.postId); - } - - @Override - public int hashCode() { - return Objects.hash(postId); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Queued.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Queued.java deleted file mode 100644 index d71019f6..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/Queued.java +++ /dev/null @@ -1,56 +0,0 @@ -package tn.mnlr.vripper.jpa.domain; - -import com.fasterxml.jackson.annotation.JsonIgnore; -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; - -import java.util.Objects; - -@Getter -@Setter -@NoArgsConstructor -@ToString -public class Queued { - - @JsonIgnore private Long id; - - private String link; - - private String threadId; - - private String postId; - - private int total = 0; - - private boolean loading = true; - - public Queued(String link, String threadId, String postId) { - this(); - this.link = link; - this.threadId = threadId; - this.postId = postId; - } - - public void done() { - this.loading = false; - } - - public void increment() { - this.total++; - } - - @Override - public boolean equals(Object o) { - if (this == o) return true; - if (o == null || getClass() != o.getClass()) return false; - Queued that = (Queued) o; - return Objects.equals(threadId, that.threadId); - } - - @Override - public int hashCode() { - return Objects.hash(threadId); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/enums/Status.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/enums/Status.java deleted file mode 100644 index 08a98b4c..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/domain/enums/Status.java +++ /dev/null @@ -1,10 +0,0 @@ -package tn.mnlr.vripper.jpa.domain.enums; - -public enum Status { - PENDING, - DOWNLOADING, - COMPLETE, - ERROR, - STOPPED, - PARTIAL -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IImageRepository.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IImageRepository.java deleted file mode 100644 index 831842f5..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IImageRepository.java +++ /dev/null @@ -1,32 +0,0 @@ -package tn.mnlr.vripper.jpa.repositories; - -import tn.mnlr.vripper.jpa.domain.Image; -import tn.mnlr.vripper.jpa.domain.enums.Status; - -import java.util.List; -import java.util.Optional; - -public interface IImageRepository extends IRepository { - - Image save(Image image); - - void deleteAllByPostId(String postId); - - List findByPostId(String postId); - - Integer countError(); - - List findByPostIdAndIsNotCompleted(String postId); - - int stopByPostIdAndIsNotCompleted(String postId); - - List findByPostIdAndIsError(String postId); - - Optional findById(Long id); - - int updateStatus(Status status, Long id); - - int updateCurrent(long current, Long id); - - int updateTotal(long total, Long id); -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/ILogEventRepository.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/ILogEventRepository.java deleted file mode 100644 index 4a175db6..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/ILogEventRepository.java +++ /dev/null @@ -1,21 +0,0 @@ -package tn.mnlr.vripper.jpa.repositories; - -import tn.mnlr.vripper.jpa.domain.LogEvent; - -import java.util.List; -import java.util.Optional; - -public interface ILogEventRepository extends IRepository { - - LogEvent save(LogEvent logEvent); - - LogEvent update(LogEvent logEvent); - - List findAll(); - - Optional findById(Long id); - - void delete(Long id); - - void deleteAll(); -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IMetadataRepository.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IMetadataRepository.java deleted file mode 100644 index 81752e5d..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IMetadataRepository.java +++ /dev/null @@ -1,14 +0,0 @@ -package tn.mnlr.vripper.jpa.repositories; - -import tn.mnlr.vripper.jpa.domain.Metadata; - -import java.util.Optional; - -public interface IMetadataRepository extends IRepository { - - Metadata save(Metadata metadata); - - Optional findByPostId(String postId); - - int deleteByPostId(String postId); -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IPostRepository.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IPostRepository.java deleted file mode 100644 index 7365774c..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IPostRepository.java +++ /dev/null @@ -1,38 +0,0 @@ -package tn.mnlr.vripper.jpa.repositories; - -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.jpa.domain.enums.Status; - -import java.util.List; -import java.util.Optional; - -public interface IPostRepository extends IRepository { - - Post save(Post post); - - Optional findByPostId(String postId); - - Optional findById(Long id); - - List findCompleted(); - - List findAll(); - - boolean existByPostId(String postId); - - int setDownloadingToStopped(); - - int deleteByPostId(String postId); - - int updateStatus(Status status, Long id); - - int updateDone(int done, Long id); - - int updateFolderName(String postFolderName, Long id); - - int updateTitle(String title, Long id); - - int updateThanked(boolean thanked, Long id); - - int updateRank(int rank, Long id); -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IQueuedRepository.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IQueuedRepository.java deleted file mode 100644 index aab60339..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IQueuedRepository.java +++ /dev/null @@ -1,20 +0,0 @@ -package tn.mnlr.vripper.jpa.repositories; - -import tn.mnlr.vripper.jpa.domain.Queued; - -import java.util.List; -import java.util.Optional; - -public interface IQueuedRepository extends IRepository { - Queued save(Queued queued); - - Optional findByThreadId(String threadId); - - List findAll(); - - Optional findById(Long id); - - int deleteByThreadId(String threadId); - - void deleteAll(); -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IRepository.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IRepository.java deleted file mode 100644 index 854550e3..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/IRepository.java +++ /dev/null @@ -1,3 +0,0 @@ -package tn.mnlr.vripper.jpa.repositories; - -public interface IRepository {} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/ImageRepository.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/ImageRepository.java deleted file mode 100644 index d88f9b04..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/ImageRepository.java +++ /dev/null @@ -1,156 +0,0 @@ -package tn.mnlr.vripper.jpa.repositories.impl; - -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.jdbc.core.JdbcTemplate; -import org.springframework.jdbc.core.RowMapper; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; -import tn.mnlr.vripper.host.Host; -import tn.mnlr.vripper.jpa.domain.Image; -import tn.mnlr.vripper.jpa.domain.enums.Status; -import tn.mnlr.vripper.jpa.repositories.IImageRepository; - -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.List; -import java.util.Optional; - -@Service -public class ImageRepository implements IImageRepository { - - private final JdbcTemplate jdbcTemplate; - private final EventBus eventBus; - - @Autowired - public ImageRepository(JdbcTemplate jdbcTemplate, EventBus eventBus) { - this.jdbcTemplate = jdbcTemplate; - this.eventBus = eventBus; - } - - private synchronized Long nextId() { - return jdbcTemplate.queryForObject("CALL NEXT VALUE FOR SEQ_IMAGE", Long.class); - } - - @Override - public Image save(Image image) { - long id = nextId(); - jdbcTemplate.update( - "INSERT INTO IMAGE (ID, CURRENT, HOST, INDEX, POST_ID, STATUS, TOTAL, URL, POST_ID_REF) VALUES (?,?,?,?,?,?,?,?,?)", - id, - image.getCurrent(), - image.getHost().getHost(), - image.getIndex(), - image.getPostId(), - image.getStatus().name(), - image.getTotal(), - image.getUrl(), - image.getPostIdRef()); - image.setId(id); - eventBus.publishEvent(Event.wrap(Event.Kind.IMAGE_UPDATE, id)); - return image; - } - - @Override - public void deleteAllByPostId(String postId) { - jdbcTemplate.update("DELETE FROM IMAGE WHERE POST_ID = ?", postId); - } - - @Override - public List findByPostId(String postId) { - return jdbcTemplate.query( - "SELECT * FROM IMAGE WHERE POST_ID = ?", new ImageRowMapper(), postId); - } - - @Override - public Integer countError() { - return jdbcTemplate.queryForObject( - "SELECT COUNT(*) FROM IMAGE AS image WHERE image.STATUS = 'ERROR'", Integer.class); - } - - @Override - public List findByPostIdAndIsNotCompleted(String postId) { - return jdbcTemplate.query( - "SELECT * FROM IMAGE AS image WHERE image.POST_ID = ? AND image.STATUS <> 'COMPLETE'", - new ImageRowMapper(), - postId); - } - - @Override - public int stopByPostIdAndIsNotCompleted(String postId) { - return jdbcTemplate.update( - "UPDATE IMAGE AS image SET image.STATUS = 'STOPPED' WHERE image.POST_ID = ? AND image.STATUS <> 'COMPLETE'", - postId); - } - - @Override - public List findByPostIdAndIsError(String postId) { - return jdbcTemplate.query( - "SELECT * FROM IMAGE AS image WHERE image.POST_ID = ? AND image.STATUS = 'ERROR'", - new ImageRowMapper(), - postId); - } - - @Override - public Optional findById(Long id) { - List images = - jdbcTemplate.query( - "SELECT * FROM IMAGE AS image WHERE image.ID = ?", new ImageRowMapper(), id); - if (images.isEmpty()) { - return Optional.empty(); - } else { - return Optional.of(images.get(0)); - } - } - - @Override - public int updateStatus(Status status, Long id) { - int mutationCount = - jdbcTemplate.update( - "UPDATE IMAGE AS image SET image.STATUS = ? WHERE image.ID = ?", status.name(), id); - eventBus.publishEvent(Event.wrap(Event.Kind.IMAGE_UPDATE, id)); - return mutationCount; - } - - @Override - public int updateCurrent(long current, Long id) { - int mutationCount = - jdbcTemplate.update( - "UPDATE IMAGE AS image SET image.CURRENT = ? WHERE image.ID = ?", current, id); - eventBus.publishEvent(Event.wrap(Event.Kind.IMAGE_UPDATE, id)); - return mutationCount; - } - - @Override - public int updateTotal(long total, Long id) { - int mutationCount = - jdbcTemplate.update( - "UPDATE IMAGE AS image SET image.TOTAL = ? WHERE image.ID = ?", total, id); - eventBus.publishEvent(Event.wrap(Event.Kind.IMAGE_UPDATE, id)); - return mutationCount; - } -} - -class ImageRowMapper implements RowMapper { - - @Override - public Image mapRow(ResultSet rs, int rowNum) throws SQLException { - Image image = new Image(); - image.setId(rs.getLong("ID")); - String host = rs.getString("HOST"); - image.setHost( - SpringContext.getBeansOfType(Host.class).values().stream() - .filter(e -> e.getHost().equals(host)) - .findAny() - .orElse(null)); - image.setUrl(rs.getString("URL")); - image.setIndex(rs.getInt("INDEX")); - image.setCurrent(rs.getLong("CURRENT")); - image.setTotal(rs.getLong("TOTAL")); - image.setStatus(Status.valueOf(rs.getString("STATUS"))); - image.setPostId(rs.getString("POST_ID")); - image.setPostIdRef(rs.getLong("POST_ID_REF")); - return image; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/LogEventRepository.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/LogEventRepository.java deleted file mode 100644 index 7e90452f..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/LogEventRepository.java +++ /dev/null @@ -1,125 +0,0 @@ -package tn.mnlr.vripper.jpa.repositories.impl; - -import lombok.extern.slf4j.Slf4j; -import org.springframework.jdbc.core.JdbcTemplate; -import org.springframework.jdbc.core.RowMapper; -import org.springframework.lang.NonNull; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; -import tn.mnlr.vripper.jpa.domain.LogEvent; -import tn.mnlr.vripper.jpa.repositories.ILogEventRepository; -import tn.mnlr.vripper.services.SettingsService; - -import java.sql.ResultSet; -import java.sql.SQLException; -import java.time.LocalDateTime; -import java.time.format.DateTimeFormatter; -import java.util.List; -import java.util.Optional; - -@Service -@Slf4j -public class LogEventRepository implements ILogEventRepository { - - private final JdbcTemplate jdbcTemplate; - private final SettingsService settingsService; - private final EventBus eventBus; - - public LogEventRepository( - JdbcTemplate jdbcTemplate, SettingsService settingsService, EventBus eventBus) { - this.jdbcTemplate = jdbcTemplate; - this.settingsService = settingsService; - this.eventBus = eventBus; - } - - private synchronized Long nextId() { - return jdbcTemplate.queryForObject("CALL NEXT VALUE FOR SEQ_EVENT", Long.class); - } - - @Override - public synchronized LogEvent save(@NonNull LogEvent logEvent) { - - int maxRecords = settingsService.getSettings().getMaxEventLog() - 1; - - Long count = jdbcTemplate.queryForObject("SELECT COUNT(*) FROM EVENT", Long.class); - if (count > maxRecords) { - List idList = - jdbcTemplate.queryForList( - "SELECT ID FROM EVENT ORDER BY TIME ASC LIMIT ?", Long.class, count - maxRecords); - idList.forEach(this::delete); - } - - long id = nextId(); - jdbcTemplate.update( - "INSERT INTO EVENT (ID, TYPE, STATUS, TIME, MESSAGE) VALUES (?,?,?,?,?)", - id, - logEvent.getType().name(), - logEvent.getStatus().name(), - logEvent.getTime().format(DateTimeFormatter.ISO_LOCAL_DATE_TIME), - logEvent.getMessage()); - logEvent.setId(id); - eventBus.publishEvent(Event.wrap(Event.Kind.LOG_EVENT_UPDATE, id)); - return logEvent; - } - - @Override - public LogEvent update(@NonNull LogEvent logEvent) { - if (logEvent.getId() == null) { - log.warn("Cannot update entity with null id"); - return logEvent; - } - - jdbcTemplate.update( - "UPDATE EVENT SET TYPE = ?, STATUS = ?, TIME = ?, MESSAGE = ? WHERE ID = ?", - logEvent.getType().name(), - logEvent.getStatus().name(), - logEvent.getTime().format(DateTimeFormatter.ISO_LOCAL_DATE_TIME), - logEvent.getMessage(), - logEvent.getId()); - eventBus.publishEvent(Event.wrap(Event.Kind.LOG_EVENT_UPDATE, logEvent.getId())); - return logEvent; - } - - @Override - public Optional findById(Long id) { - List logEvents = - jdbcTemplate.query("SELECT * FROM EVENT WHERE ID = ?", new LogEventRowMapper(), id); - if (logEvents.isEmpty()) { - return Optional.empty(); - } else { - return Optional.of(logEvents.get(0)); - } - } - - @Override - public List findAll() { - return jdbcTemplate.query("SELECT * FROM EVENT", new LogEventRowMapper()); - } - - @Override - public void delete(Long id) { - jdbcTemplate.update("DELETE FROM EVENT WHERE ID = ?", id); - eventBus.publishEvent(Event.wrap(Event.Kind.LOG_EVENT_REMOVE, id)); - } - - @Override - public void deleteAll() { - jdbcTemplate.update("DELETE FROM EVENT"); - } -} - -class LogEventRowMapper implements RowMapper { - - @Override - public LogEvent mapRow(ResultSet rs, int rowNum) throws SQLException { - LogEvent logEvent = new LogEvent(); - logEvent.setId(rs.getLong("ID")); - logEvent.setType(LogEvent.Type.valueOf(rs.getString("TYPE"))); - logEvent.setStatus(LogEvent.Status.valueOf(rs.getString("STATUS"))); - logEvent.setTime( - LocalDateTime.parse(rs.getString("TIME"), DateTimeFormatter.ISO_LOCAL_DATE_TIME)); - logEvent.setMessage(rs.getString("MESSAGE")); - return logEvent; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/MetadataRepository.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/MetadataRepository.java deleted file mode 100644 index 82d6a145..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/MetadataRepository.java +++ /dev/null @@ -1,76 +0,0 @@ -package tn.mnlr.vripper.jpa.repositories.impl; - -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.jdbc.core.JdbcTemplate; -import org.springframework.jdbc.core.RowMapper; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; -import tn.mnlr.vripper.jpa.domain.Metadata; -import tn.mnlr.vripper.jpa.repositories.IMetadataRepository; - -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.List; -import java.util.Optional; - -@Service -public class MetadataRepository implements IMetadataRepository { - - private final JdbcTemplate jdbcTemplate; - private final EventBus eventBus; - - @Autowired - public MetadataRepository(JdbcTemplate jdbcTemplate, EventBus eventBus) { - this.jdbcTemplate = jdbcTemplate; - this.eventBus = eventBus; - } - - @Override - public Metadata save(Metadata metadata) { - jdbcTemplate.update( - "INSERT INTO METADATA (POST_ID_REF, POST_ID, POSTED_BY, RESOLVED_NAMES) VALUES (?,?,?,?)", - metadata.getPostIdRef(), - metadata.getPostId(), - metadata.getPostedBy(), - String.join("%sep%", metadata.getResolvedNames())); - eventBus.publishEvent(Event.wrap(Event.Kind.METADATA_UPDATE, metadata.getPostIdRef())); - return metadata; - } - - @Override - public Optional findByPostId(String postId) { - List metadata = - jdbcTemplate.query( - "SELECT metadata.* FROM METADATA AS metadata WHERE metadata.POST_ID = ?", - new MetadataRowMapper(), - postId); - if (metadata.isEmpty()) { - return Optional.empty(); - } else { - return Optional.of(metadata.get(0)); - } - } - - @Override - public int deleteByPostId(String postId) { - return jdbcTemplate.update( - "DELETE FROM METADATA AS metadata WHERE metadata.POST_ID = ?", postId); - } -} - -class MetadataRowMapper implements RowMapper { - - @Override - public Metadata mapRow(ResultSet rs, int rowNum) throws SQLException { - Metadata metadata = new Metadata(); - metadata.setPostIdRef(rs.getLong("POST_ID_REF")); - metadata.setPostId(rs.getString("POST_ID")); - metadata.setPostedBy(rs.getString("POSTED_BY")); - String resolvedNames = rs.getString("RESOLVED_NAMES"); - if (resolvedNames != null && !resolvedNames.isBlank()) { - metadata.setResolvedNames(List.of(resolvedNames.split("%sep%"))); - } - return metadata; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/PostRepository.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/PostRepository.java deleted file mode 100644 index d77293b9..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/PostRepository.java +++ /dev/null @@ -1,230 +0,0 @@ -package tn.mnlr.vripper.jpa.repositories.impl; - -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.jdbc.core.JdbcTemplate; -import org.springframework.jdbc.core.RowMapper; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; -import tn.mnlr.vripper.jpa.domain.Metadata; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.jpa.domain.enums.Status; -import tn.mnlr.vripper.jpa.repositories.IPostRepository; - -import java.sql.ResultSet; -import java.sql.SQLException; -import java.sql.Timestamp; -import java.util.List; -import java.util.Optional; -import java.util.Set; - -@Service -public class PostRepository implements IPostRepository { - - private final JdbcTemplate jdbcTemplate; - private final EventBus eventBus; - - @Autowired - public PostRepository(JdbcTemplate jdbcTemplate, EventBus eventBus) { - this.jdbcTemplate = jdbcTemplate; - this.eventBus = eventBus; - } - - private synchronized Long nextId() { - return jdbcTemplate.queryForObject("CALL NEXT VALUE FOR SEQ_POST", Long.class); - } - - @Override - public Post save(Post post) { - long id = nextId(); - jdbcTemplate.update( - "INSERT INTO POST (ID, DONE, FORUM, HOSTS, POST_FOLDER_NAME, POST_ID, PREVIEWS, SECURITY_TOKEN, STATUS, THANKED, THREAD_ID, THREAD_TITLE, TITLE, TOTAL, URL, ADDED_ON, RANK) VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)", - id, - post.getDone(), - post.getForum(), - String.join(";", post.getHosts()), - post.getDownloadDirectory(), - post.getPostId(), - String.join(";", post.getPreviews()), - post.getSecurityToken(), - post.getStatus().name(), - post.isThanked(), - post.getThreadId(), - post.getThreadTitle(), - post.getTitle(), - post.getTotal(), - post.getUrl(), - Timestamp.valueOf(post.getAddedOn()), - post.getRank()); - post.setId(id); - eventBus.publishEvent(Event.wrap(Event.Kind.POST_UPDATE, id)); - return post; - } - - @Override - public Optional findByPostId(String postId) { - List posts = - jdbcTemplate.query( - "SELECT metadata.*,post.* FROM METADATA metadata FULL JOIN POST post ON metadata.POST_ID_REF = post.ID WHERE post.POST_ID = ?", - new PostRowMapper(), - postId); - if (posts.isEmpty()) { - return Optional.empty(); - } else { - return Optional.of(posts.get(0)); - } - } - - @Override - public List findCompleted() { - return jdbcTemplate.query( - "SELECT POST_ID FROM POST AS post WHERE status = 'COMPLETE' AND done >= total", - ((rs, rowNum) -> rs.getString("POST_ID"))); - } - - @Override - public Optional findById(Long id) { - List posts = - jdbcTemplate.query( - "SELECT metadata.*,post.* FROM METADATA metadata FULL JOIN POST post ON metadata.POST_ID_REF = post.ID WHERE post.ID = ?", - new PostRowMapper(), - id); - if (posts.isEmpty()) { - return Optional.empty(); - } else { - return Optional.of(posts.get(0)); - } - } - - @Override - public List findAll() { - return jdbcTemplate.query( - "SELECT metadata.*,post.* FROM METADATA metadata FULL JOIN POST post ON metadata.POST_ID_REF = post.ID", - new PostRowMapper()); - } - - @Override - public boolean existByPostId(String postId) { - Integer count = - jdbcTemplate.queryForObject( - "SELECT COUNT(*) FROM POST AS post WHERE post.POST_ID = ?", Integer.class, postId); - if (count == null) { - return false; - } else { - return count > 0; - } - } - - @Override - public int setDownloadingToStopped() { - return jdbcTemplate.update( - "UPDATE POST AS post SET post.STATUS = 'STOPPED' WHERE post.STATUS = 'DOWNLOADING' OR post.STATUS = 'PARTIAL' OR post.STATUS = 'PENDING'"); - } - - @Override - public int deleteByPostId(String postId) { - int mutationCount = - jdbcTemplate.update("DELETE FROM POST AS post WHERE post.POST_ID = ?", postId); - eventBus.publishEvent(Event.wrap(Event.Kind.POST_REMOVE, postId)); - return mutationCount; - } - - @Override - public int updateStatus(Status status, Long id) { - int mutationCount = - jdbcTemplate.update( - "UPDATE POST AS post SET post.STATUS = ? WHERE post.ID = ?", status.name(), id); - eventBus.publishEvent(Event.wrap(Event.Kind.POST_UPDATE, id)); - return mutationCount; - } - - @Override - public int updateDone(int done, Long id) { - int mutationCount = - jdbcTemplate.update("UPDATE POST AS post SET post.DONE = ? WHERE post.ID = ?", done, id); - eventBus.publishEvent(Event.wrap(Event.Kind.POST_UPDATE, id)); - return mutationCount; - } - - @Override - public int updateFolderName(String postFolderName, Long id) { - int mutationCount = - jdbcTemplate.update( - "UPDATE POST AS post SET post.POST_FOLDER_NAME = ? WHERE post.ID = ?", - postFolderName, - id); - eventBus.publishEvent(Event.wrap(Event.Kind.POST_UPDATE, id)); - return mutationCount; - } - - @Override - public int updateTitle(String title, Long id) { - int mutationCount = - jdbcTemplate.update("UPDATE POST AS post SET post.TITLE = ? WHERE post.ID = ?", title, id); - eventBus.publishEvent(Event.wrap(Event.Kind.POST_UPDATE, id)); - return mutationCount; - } - - @Override - public int updateThanked(boolean thanked, Long id) { - int mutationCount = - jdbcTemplate.update( - "UPDATE POST AS post SET post.THANKED = ? WHERE post.ID = ?", thanked, id); - eventBus.publishEvent(Event.wrap(Event.Kind.POST_UPDATE, id)); - return mutationCount; - } - - @Override - public int updateRank(int rank, Long id) { - int mutationCount = - jdbcTemplate.update("UPDATE POST AS post SET post.RANK = ? WHERE post.ID = ?", rank, id); - eventBus.publishEvent(Event.wrap(Event.Kind.POST_UPDATE, id)); - return mutationCount; - } -} - -class PostRowMapper implements RowMapper { - - private static final String DELIMITER = ";"; - - @Override - public Post mapRow(ResultSet rs, int rowNum) throws SQLException { - - Post post = new Post(); - post.setId(rs.getLong("post.ID")); - post.setStatus(Status.valueOf(rs.getString("post.STATUS"))); - post.setPostId(rs.getString("post.POST_ID")); - post.setThreadTitle(rs.getString("post.THREAD_TITLE")); - post.setThreadId(rs.getString("post.THREAD_ID")); - post.setTitle(rs.getString("post.TITLE")); - post.setUrl(rs.getString("post.URL")); - post.setDone(rs.getInt("post.DONE")); - post.setTotal(rs.getInt("post.TOTAL")); - post.setHosts(Set.of(rs.getString("post.HOSTS").split(DELIMITER))); - post.setForum(rs.getString("post.FORUM")); - post.setSecurityToken(rs.getString("post.SECURITY_TOKEN")); - post.setDownloadDirectory(rs.getString("post.POST_FOLDER_NAME")); - post.setThanked(rs.getBoolean("post.THANKED")); - String previews; - if ((previews = rs.getString("post.PREVIEWS")) != null) { - post.setPreviews(Set.of(previews.split(DELIMITER))); - } - post.setAddedOn(rs.getTimestamp("post.ADDED_ON").toLocalDateTime()); - post.setRank(rs.getInt("post.RANK")); - - Long metadataId = rs.getLong("metadata.POST_ID_REF"); - if (!rs.wasNull()) { - Metadata metadata = new Metadata(); - metadata.setPostIdRef(metadataId); - metadata.setPostId(rs.getString("metadata.POST_ID")); - String resolvedNames = rs.getString("metadata.RESOLVED_NAMES"); - if (resolvedNames != null && !resolvedNames.isBlank()) { - metadata.setResolvedNames(List.of(resolvedNames.split("%sep%"))); - } - metadata.setPostedBy(rs.getString("metadata.POSTED_BY")); - post.setMetadata(metadata); - } - - return post; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/QueuedRepository.java b/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/QueuedRepository.java deleted file mode 100644 index f314bcd9..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/jpa/repositories/impl/QueuedRepository.java +++ /dev/null @@ -1,107 +0,0 @@ -package tn.mnlr.vripper.jpa.repositories.impl; - -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.jdbc.core.JdbcTemplate; -import org.springframework.jdbc.core.RowMapper; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; -import tn.mnlr.vripper.jpa.domain.Queued; -import tn.mnlr.vripper.jpa.repositories.IQueuedRepository; - -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.List; -import java.util.Optional; - -@Service -public class QueuedRepository implements IQueuedRepository { - - private final JdbcTemplate jdbcTemplate; - private final EventBus eventBus; - - @Autowired - public QueuedRepository(JdbcTemplate jdbcTemplate, EventBus eventBus) { - this.jdbcTemplate = jdbcTemplate; - this.eventBus = eventBus; - } - - private synchronized Long nextId() { - return jdbcTemplate.queryForObject("CALL NEXT VALUE FOR SEQ_QUEUED", Long.class); - } - - @Override - public Queued save(Queued queued) { - long id = nextId(); - jdbcTemplate.update( - "INSERT INTO QUEUED (ID, TOTAL, LINK, LOADING, POST_ID, THREAD_ID) values (?,?,?,?,?,?)", - id, - queued.getTotal(), - queued.getLink(), - queued.isLoading(), - queued.getPostId(), - queued.getThreadId()); - queued.setId(id); - eventBus.publishEvent(Event.wrap(Event.Kind.QUEUED_UPDATE, id)); - return queued; - } - - @Override - public Optional findByThreadId(String threadId) { - List queuedList = - jdbcTemplate.query( - "SELECT * FROM QUEUED AS queued WHERE queued.THREAD_ID = ?", - new QueuedRowMapper(), - threadId); - if (queuedList.isEmpty()) { - return Optional.empty(); - } else { - return Optional.of(queuedList.get(0)); - } - } - - @Override - public List findAll() { - return jdbcTemplate.query("SELECT * FROM QUEUED", new QueuedRowMapper()); - } - - @Override - public Optional findById(Long id) { - List queuedList = - jdbcTemplate.query( - "SELECT * FROM QUEUED AS queued WHERE queued.ID = ?", new QueuedRowMapper(), id); - if (queuedList.isEmpty()) { - return Optional.empty(); - } else { - return Optional.of(queuedList.get(0)); - } - } - - @Override - public int deleteByThreadId(String threadId) { - int mutationCount = - jdbcTemplate.update("DELETE FROM QUEUED AS queued WHERE THREAD_ID = ?", threadId); - eventBus.publishEvent(Event.wrap(Event.Kind.QUEUED_REMOVE, threadId)); - return mutationCount; - } - - @Override - public void deleteAll() { - jdbcTemplate.update("DELETE FROM QUEUED"); - } -} - -class QueuedRowMapper implements RowMapper { - - @Override - public Queued mapRow(ResultSet rs, int rowNum) throws SQLException { - Queued queued = new Queued(); - queued.setId(rs.getLong("ID")); - queued.setLink(rs.getString("LINK")); - queued.setThreadId(rs.getString("THREAD_ID")); - queued.setPostId(rs.getString("POST_ID")); - queued.setTotal(rs.getInt("TOTAL")); - queued.setLoading(rs.getBoolean("LOADING")); - return queued; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/ConnectionService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/ConnectionService.java deleted file mode 100644 index c4f9952d..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/ConnectionService.java +++ /dev/null @@ -1,154 +0,0 @@ -package tn.mnlr.vripper.services; - -import lombok.Getter; -import lombok.extern.slf4j.Slf4j; -import net.jodah.failsafe.RetryPolicy; -import org.apache.http.client.config.CookieSpecs; -import org.apache.http.client.config.RequestConfig; -import org.apache.http.client.methods.AbstractExecutionAwareRequest; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.client.protocol.HttpClientContext; -import org.apache.http.impl.client.HttpClientBuilder; -import org.apache.http.impl.client.HttpClients; -import org.apache.http.impl.client.LaxRedirectStrategy; -import org.apache.http.impl.conn.PoolingHttpClientConnectionManager; -import org.springframework.scheduling.annotation.EnableScheduling; -import org.springframework.scheduling.annotation.Scheduled; -import org.springframework.stereotype.Service; -import reactor.core.Disposable; -import tn.mnlr.vripper.download.DownloadJob; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; -import tn.mnlr.vripper.services.domain.Settings; - -import javax.annotation.PreDestroy; -import java.net.URI; -import java.time.temporal.ChronoUnit; -import java.util.List; -import java.util.concurrent.TimeUnit; - -@Service -@EnableScheduling -@Slf4j -public class ConnectionService { - - private final Disposable disposable; - private PoolingHttpClientConnectionManager pcm; - private RequestConfig rc; - - @Getter private RetryPolicy retryPolicy; - - private int connectionTimeout; - private int maxAttempts; - - public ConnectionService(EventBus eventBus, SettingsService settingsService) { - connectionTimeout = settingsService.getSettings().getConnectionTimeout(); - maxAttempts = settingsService.getSettings().getMaxAttempts(); - disposable = - eventBus - .flux() - .filter(e -> e.getKind().equals(Event.Kind.SETTINGS_UPDATE)) - .map(e -> ((Settings) e.getData())) - .subscribe( - settings -> { - if (connectionTimeout != settings.getConnectionTimeout()) { - connectionTimeout = settings.getConnectionTimeout(); - buildRequestConfig(); - } - if (maxAttempts != settings.getMaxAttempts()) { - maxAttempts = settings.getMaxAttempts(); - buildRetryPolicy(); - } - }); - - buildRequestConfig(); - buildRetryPolicy(); - buildConnectionPool(); - } - - @PreDestroy - private void destroy() { - disposable.dispose(); - } - - private void buildRetryPolicy() { - retryPolicy = - new RetryPolicy<>() - .withDelay(2, 5, ChronoUnit.SECONDS) - .withMaxAttempts(maxAttempts) - .onFailedAttempt( - e -> - log.warn( - String.format("#%d tries failed", e.getAttemptCount()), - e.getLastFailure())); - } - - private void buildConnectionPool() { - pcm = new PoolingHttpClientConnectionManager(); - pcm.setMaxTotal(Integer.MAX_VALUE); - pcm.setDefaultMaxPerRoute(Integer.MAX_VALUE); - } - - private void buildRequestConfig() { - rc = - RequestConfig.custom() - .setConnectionRequestTimeout(connectionTimeout * 1000) - .setConnectTimeout(connectionTimeout * 1000) - .setSocketTimeout(connectionTimeout * 1000) - .setCookieSpec(CookieSpecs.STANDARD) - .build(); - } - - @Scheduled(fixedDelay = 5_000) - private void idleConnectionMonitoring() { - pcm.closeExpiredConnections(); - pcm.closeIdleConnections(30, TimeUnit.SECONDS); - } - - public HttpClientBuilder getClient() { - return HttpClients.custom() - .setConnectionManager(pcm) - .setRedirectStrategy(new LaxRedirectStrategy()) - .disableAutomaticRetries() - .setDefaultRequestConfig(rc); - } - - public HttpGet buildHttpGet(String url, final HttpClientContext context) { - HttpGet httpGet = new HttpGet(url.replace(" ", "+")); - httpGet.addHeader( - "User-Agent", - "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36"); - addToContext(context, httpGet); - return httpGet; - } - - public HttpPost buildHttpPost(String url, final HttpClientContext context) { - HttpPost httpPost = new HttpPost(url.replace(" ", "+")); - httpPost.addHeader( - "User-Agent", - "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36"); - addToContext(context, httpPost); - return httpPost; - } - - public HttpGet buildHttpGet(URI uri, final HttpClientContext context) { - HttpGet httpGet = new HttpGet(uri); - httpGet.addHeader( - "User-Agent", - "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36"); - addToContext(context, httpGet); - return httpGet; - } - - public void addToContext(HttpClientContext context, AbstractExecutionAwareRequest request) { - if (context != null) { - List requests = - (List) - context.getAttribute(DownloadJob.ContextAttributes.OPEN_CONNECTION.toString()); - if (requests != null) { - requests.add(request); - } - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/DataService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/DataService.java deleted file mode 100644 index df398066..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/DataService.java +++ /dev/null @@ -1,244 +0,0 @@ -package tn.mnlr.vripper.services; - -import lombok.NonNull; -import lombok.extern.slf4j.Slf4j; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.springframework.transaction.annotation.Transactional; -import tn.mnlr.vripper.jpa.domain.*; -import tn.mnlr.vripper.jpa.domain.enums.Status; -import tn.mnlr.vripper.jpa.repositories.IImageRepository; -import tn.mnlr.vripper.jpa.repositories.IMetadataRepository; -import tn.mnlr.vripper.jpa.repositories.IPostRepository; -import tn.mnlr.vripper.jpa.repositories.IQueuedRepository; -import tn.mnlr.vripper.jpa.repositories.impl.LogEventRepository; - -import javax.annotation.PostConstruct; -import java.util.*; -import java.util.stream.Collectors; - -@Service -@Transactional -@Slf4j -public class DataService { - - private final IPostRepository postRepository; - private final IImageRepository imageRepository; - private final IQueuedRepository queuedRepository; - private final IMetadataRepository metadataRepository; - private final SettingsService settingsService; - private final LogEventRepository eventRepository; - - @Autowired - public DataService( - IPostRepository postRepository, - IImageRepository imageRepository, - IQueuedRepository queuedRepository, - IMetadataRepository metadataRepository, - SettingsService settingsService, - LogEventRepository eventRepository) { - this.postRepository = postRepository; - this.imageRepository = imageRepository; - this.queuedRepository = queuedRepository; - this.metadataRepository = metadataRepository; - this.settingsService = settingsService; - this.eventRepository = eventRepository; - } - - private void save(Post post) { - postRepository.save(post); - } - - private void save(Queued queued) { - queuedRepository.save(queued); - } - - private void save(Image image) { - imageRepository.save(image); - } - - public boolean exists(String postId) { - return postRepository.existByPostId(postId); - } - - public void newPost(Post post, Collection images) { - save(post); - images.forEach( - image -> { - image.setPostIdRef(post.getId()); - save(image); - }); - sortPostsByRank(); - } - - public synchronized void afterJobFinish(Image image, Post post) { - if (image.getStatus().equals(Status.COMPLETE)) { - post.setDone(post.getDone() + 1); - updatePostDone(post.getDone(), post.getId()); - } else if (image.getStatus().equals(Status.ERROR)) { - post.setStatus(Status.PARTIAL); - updatePostStatus(post.getStatus(), post.getId()); - } - } - - private void updatePostDone(int done, Long id) { - postRepository.updateDone(done, id); - } - - public void updatePostStatus(Status status, Long id) { - postRepository.updateStatus(status, id); - } - - public void updatePostThanks(boolean thanked, Long id) { - postRepository.updateThanked(thanked, id); - } - - public void updatePostTitle(String title, Long id) { - postRepository.updateTitle(title, id); - } - - public void updatePostDownloadDirectory(String postFolderName, Long id) { - postRepository.updateFolderName(postFolderName, id); - } - - public void updatePostRank(int rank, Long id) { - postRepository.updateRank(rank, id); - } - - public void finishPost(@NonNull Post post) { - if (!imageRepository.findByPostIdAndIsError(post.getPostId()).isEmpty()) { - post.setStatus(Status.ERROR); - updatePostStatus(post.getStatus(), post.getId()); - } else { - if (post.getDone() < post.getTotal()) { - post.setStatus(Status.STOPPED); - updatePostStatus(post.getStatus(), post.getId()); - } else { - post.setStatus(Status.COMPLETE); - updatePostStatus(post.getStatus(), post.getId()); - if (settingsService.getSettings().getClearCompleted()) { - remove(List.of(post.getPostId())); - } - } - } - } - - private void remove(@NonNull final List postIds) { - for (String postId : postIds) { - imageRepository.deleteAllByPostId(postId); - metadataRepository.deleteByPostId(postId); - postRepository.deleteByPostId(postId); - } - sortPostsByRank(); - } - - public void newQueueLink(@NonNull final Queued queued) { - save(queued); - } - - public void removeQueueLink(@NonNull final String threadId) { - queuedRepository.deleteByThreadId(threadId); - } - - public List clearCompleted() { - List completed = postRepository.findCompleted(); - remove(completed); - return completed; - } - - public void removeAll(final List postIds) { - - remove( - Objects.requireNonNullElse( - postIds, - postRepository.findAll().stream().map(Post::getPostId).collect(Collectors.toList()))); - } - - public List findByPostIdAndIsNotCompleted(@NonNull String postId) { - return imageRepository.findByPostIdAndIsNotCompleted(postId); - } - - public long countErrorImages() { - return imageRepository.countError(); - } - - public List findImagesByPostId(String postId) { - return imageRepository.findByPostId(postId); - } - - public List findAllPosts() { - return postRepository.findAll(); - } - - public Optional findPostByPostId(String postId) { - return postRepository.findByPostId(postId); - } - - public void stopImagesByPostIdAndIsNotCompleted(String postId) { - imageRepository.stopByPostIdAndIsNotCompleted(postId); - } - - public Optional findQueuedByThreadId(String threadId) { - return queuedRepository.findByThreadId(threadId); - } - - public List findAllQueued() { - return queuedRepository.findAll(); - } - - public Optional findById(Long aLong) { - return postRepository.findById(aLong); - } - - public Optional findImageById(Long aLong) { - return imageRepository.findById(aLong); - } - - public Optional findQueuedById(Long aLong) { - return queuedRepository.findById(aLong); - } - - public synchronized void setMetadata(Post post, Metadata metadata) { - if (metadataRepository.findByPostId(post.getPostId()).isEmpty()) { - metadata.setPostIdRef(post.getId()); - metadataRepository.save(metadata); - } - } - - public Optional findMetadataByPostId(String postId) { - return metadataRepository.findByPostId(postId); - } - - public void updateImageStatus(Status status, Long id) { - imageRepository.updateStatus(status, id); - } - - public void updateImageCurrent(long current, Long id) { - imageRepository.updateCurrent(current, id); - } - - public void updateImageTotal(long total, Long id) { - imageRepository.updateTotal(total, id); - } - - public Optional findEventById(Long id) { - return eventRepository.findById(id); - } - - public List findAllEvents() { - return eventRepository.findAll(); - } - - public void clearQueueLinks() { - queuedRepository.deleteAll(); - } - - public synchronized void sortPostsByRank() { - List posts = findAllPosts(); - posts.sort(Comparator.comparing(Post::getAddedOn)); - for (int i = 0; i < posts.size(); i++) { - posts.get(i).setRank(i); - updatePostRank(i, posts.get(i).getId()); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/DownloadSpeedService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/DownloadSpeedService.java deleted file mode 100644 index be24d15f..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/DownloadSpeedService.java +++ /dev/null @@ -1,41 +0,0 @@ -package tn.mnlr.vripper.services; - -import lombok.Getter; -import org.springframework.scheduling.annotation.EnableScheduling; -import org.springframework.scheduling.annotation.Scheduled; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; - -import java.util.concurrent.atomic.AtomicLong; - -@Service -@EnableScheduling -public class DownloadSpeedService { - - private final AtomicLong read = new AtomicLong(0); - private final EventBus eventBus; - @Getter private long currentValue; - private boolean allowWrite = false; - - public DownloadSpeedService(EventBus eventBus) { - this.eventBus = eventBus; - } - - public void increase(long read) { - if (allowWrite) { - this.read.addAndGet(read); - } - } - - @Scheduled(fixedDelay = 1000) - private void calc() { - allowWrite = false; - long newValue = read.getAndSet(0); - if (newValue != currentValue) { - currentValue = newValue; - eventBus.publishEvent(Event.wrap(Event.Kind.BYTES_PER_SECOND, currentValue)); - } - allowWrite = true; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/GlobalStateService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/GlobalStateService.java deleted file mode 100644 index 73aef1e0..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/GlobalStateService.java +++ /dev/null @@ -1,42 +0,0 @@ -package tn.mnlr.vripper.services; - -import lombok.Getter; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.scheduling.annotation.EnableScheduling; -import org.springframework.scheduling.annotation.Scheduled; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.download.DownloadService; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; -import tn.mnlr.vripper.services.domain.GlobalState; - -@Service -@EnableScheduling -public class GlobalStateService { - - private final DownloadService downloadService; - private final DataService dataService; - private final EventBus eventBus; - @Getter private GlobalState currentState; - - @Autowired - public GlobalStateService( - DownloadService downloadService, DataService dataService, EventBus eventBus) { - this.downloadService = downloadService; - this.dataService = dataService; - this.eventBus = eventBus; - } - - @Scheduled(fixedDelay = 3000) - private void interval() { - GlobalState newGlobalState = - new GlobalState( - downloadService.runningCount(), - downloadService.pendingCount(), - dataService.countErrorImages()); - if (!newGlobalState.equals(currentState)) { - currentState = newGlobalState; - eventBus.publishEvent(Event.wrap(Event.Kind.GLOBAL_STATE, currentState)); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/HostService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/HostService.java deleted file mode 100644 index 1a997df6..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/HostService.java +++ /dev/null @@ -1,113 +0,0 @@ -package tn.mnlr.vripper.services; - -import lombok.Getter; -import lombok.extern.slf4j.Slf4j; -import org.apache.http.Header; -import org.apache.http.client.HttpClient; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.client.protocol.HttpClientContext; -import org.apache.http.util.EntityUtils; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import tn.mnlr.vripper.exception.HostException; -import tn.mnlr.vripper.exception.HtmlProcessorException; - -import java.io.IOException; -import java.net.URI; -import java.net.URISyntaxException; - -@Service -@Slf4j -public class HostService { - - private final ConnectionService cm; - private final HtmlProcessorService htmlProcessorService; - - @Autowired - public HostService(ConnectionService cm, HtmlProcessorService htmlProcessorService) { - this.cm = cm; - this.htmlProcessorService = htmlProcessorService; - } - - public Response getResponse(final String url, final HttpClientContext context) - throws HostException { - String basePage; - - HttpClient client = cm.getClient().build(); - HttpGet httpGet = cm.buildHttpGet(url, context); - Header[] headers; - log.debug(String.format("Requesting %s", url)); - try (CloseableHttpResponse response = - (CloseableHttpResponse) client.execute(httpGet, context)) { - if (response.getStatusLine().getStatusCode() / 100 != 2) { - throw new HostException( - String.format( - "Unexpected response code: %d", response.getStatusLine().getStatusCode())); - } - headers = response.getAllHeaders(); - basePage = EntityUtils.toString(response.getEntity()); - log.debug(String.format("%s response: %n%s", url, basePage)); - EntityUtils.consumeQuietly(response.getEntity()); - } catch (IOException e) { - throw new HostException(e); - } - - try { - log.debug(String.format("Cleaning %s response", url)); - return new Response(htmlProcessorService.clean(basePage), headers); - } catch (HtmlProcessorException e) { - throw new HostException(e); - } - } - - public String appendUri(String uri, String appendQuery) throws URISyntaxException { - URI oldUri = new URI(uri); - - String newQuery = oldUri.getQuery(); - if (newQuery == null) { - newQuery = appendQuery; - } else { - newQuery += "&" + appendQuery; - } - - return new URI( - oldUri.getScheme(), - oldUri.getAuthority(), - oldUri.getPath(), - newQuery, - oldUri.getFragment()) - .toString(); - } - - public String getDefaultImageName(final String imgUrl) { - String imageTitle = imgUrl.substring(imgUrl.lastIndexOf('/') + 1); - log.debug(String.format("Extracting name from url %s: %s", imgUrl, imageTitle)); - return imgUrl; - } - - @Getter - public static class Response { - - private final Document document; - private final Header[] headers; - - protected Response(Document document, Header[] headers) { - this.document = document; - this.headers = headers; - } - } - - @Getter - public static class NameUrl { - - private final String name; - private final String url; - - public NameUrl(String name, String url) { - this.name = name; - this.url = url; - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/HtmlProcessorService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/HtmlProcessorService.java deleted file mode 100644 index 21576e81..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/HtmlProcessorService.java +++ /dev/null @@ -1,22 +0,0 @@ -package tn.mnlr.vripper.services; - -import org.htmlcleaner.CleanerProperties; -import org.htmlcleaner.DomSerializer; -import org.htmlcleaner.HtmlCleaner; -import org.htmlcleaner.TagNode; -import org.springframework.stereotype.Service; -import org.w3c.dom.Document; -import tn.mnlr.vripper.exception.HtmlProcessorException; - -@Service -public class HtmlProcessorService { - - public Document clean(String htmlContent) throws HtmlProcessorException { - try { - TagNode clean = new HtmlCleaner().clean(htmlContent); - return new DomSerializer(new CleanerProperties()).createDOM(clean); - } catch (Exception e) { - throw new HtmlProcessorException(e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/MetadataService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/MetadataService.java deleted file mode 100644 index fd52ec7d..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/MetadataService.java +++ /dev/null @@ -1,50 +0,0 @@ -package tn.mnlr.vripper.services; - -import lombok.extern.slf4j.Slf4j; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.tasks.MetadataRunnable; - -import java.util.ArrayList; -import java.util.List; -import java.util.Map; -import java.util.concurrent.ConcurrentHashMap; - -@Slf4j -@Service -public class MetadataService { - - private final Map fetchingMetadata = new ConcurrentHashMap<>(); - private final ThreadPoolService threadPoolService; - - public MetadataService(ThreadPoolService threadPoolService) { - this.threadPoolService = threadPoolService; - } - - public void startFetchingMetadata(Post post) { - MetadataRunnable runnable = new MetadataRunnable(post); - threadPoolService.getGeneralExecutor().submit(runnable); - fetchingMetadata.put(post.getPostId(), runnable); - } - - public void stopFetchingMetadata(List postIds) { - List stopping = new ArrayList<>(); - - for (Map.Entry entry : this.fetchingMetadata.entrySet()) { - if (postIds.contains(entry.getKey())) { - stopping.add(entry.getValue()); - entry.getValue().stop(); - } - } - - while (!stopping.isEmpty()) { - stopping.removeIf(MetadataRunnable::isFinished); - try { - Thread.sleep(100); - } catch (InterruptedException e) { - Thread.currentThread().interrupt(); - } - } - postIds.forEach(fetchingMetadata::remove); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/PathService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/PathService.java deleted file mode 100644 index 8184028c..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/PathService.java +++ /dev/null @@ -1,126 +0,0 @@ -package tn.mnlr.vripper.services; - -import lombok.Getter; -import lombok.NonNull; -import lombok.extern.slf4j.Slf4j; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.exception.RenameException; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.services.domain.Settings; - -import java.io.File; -import java.io.IOException; -import java.nio.file.Files; -import java.nio.file.StandardCopyOption; -import java.util.concurrent.locks.ReentrantLock; - -@Service -@Slf4j -public class PathService { - - private final DataService dataService; - - @Getter private final ReentrantLock directoryAccess = new ReentrantLock(); - - public PathService(DataService dataService) { - this.dataService = dataService; - } - - public final File calcDownloadDirectory(Post post, Settings settings) { - return new File(settings.getDownloadPath(), post.getDownloadDirectory()); - } - - private File getRootFolder( - @NonNull String forum, @NonNull String threadTitle, Settings settings) { - File sourceFolder = - settings.getSubLocation() - ? new File(settings.getDownloadPath(), sanitize(forum)) - : new File(settings.getDownloadPath()); - return settings.getThreadSubLocation() ? new File(sourceFolder, threadTitle) : sourceFolder; - } - - public final void createDefaultPostFolder(Post post, Settings settings) { - File downloadDirectory = getRootFolder(post.getForum(), post.getThreadTitle(), settings); - downloadDirectory = - new File( - downloadDirectory, - settings.getAppendPostId() - ? sanitize(post.getTitle()) + "_" + post.getPostId() - : sanitize(post.getTitle())); - downloadDirectory = makeDir(downloadDirectory); - post.setDownloadDirectory( - downloadDirectory.getAbsolutePath().replace(settings.getDownloadPath(), "")); - dataService.updatePostDownloadDirectory(post.getDownloadDirectory(), post.getId()); - } - - public final void rename(@NonNull String postId, @NonNull String altName, Settings settings) - throws RenameException { - Post post = dataService.findPostByPostId(postId).orElseThrow(); - if (altName.equals(post.getTitle())) { - return; - } - - // Download have not started yet - if (post.getDownloadDirectory() == null) { - post.setTitle(altName); - dataService.updatePostTitle(post.getTitle(), post.getId()); - return; - } - - File newDownloadDirectory = getRootFolder(post.getForum(), post.getThreadTitle(), settings); - newDownloadDirectory = new File(newDownloadDirectory, sanitize(altName)); - File currentDownloadDirectory = calcDownloadDirectory(post, settings); - try { - directoryAccess.lock(); - Files.move( - currentDownloadDirectory.toPath(), - newDownloadDirectory.toPath(), - StandardCopyOption.ATOMIC_MOVE); - post.setDownloadDirectory( - newDownloadDirectory.getAbsolutePath().replace(settings.getDownloadPath(), "")); - dataService.updatePostDownloadDirectory(post.getDownloadDirectory(), post.getId()); - - post.setTitle(altName); - dataService.updatePostTitle(post.getTitle(), post.getId()); - } catch (IOException e) { - throw new RenameException( - String.format( - "Failed to move files from %s to %s", currentDownloadDirectory, newDownloadDirectory), - e); - } finally { - directoryAccess.unlock(); - } - } - - private File makeDir(@NonNull final File sourceFolder) { - int counter = 1; - File folder = sourceFolder; - - while (folder.exists()) { - folder = new File(sourceFolder.toString() + '.' + counter++); - } - - if (!folder.mkdirs()) { - throw new RuntimeException(String.format("Failed to create the folder %s", sourceFolder)); - } - - return folder; - } - - private String sanitize(final String folderName) { - String sanitizedFolderName = - folderName.replaceAll("\\.|\\\\|/|\\||:|\\?|\\*|\"|<|>|\\p{Cntrl}", "_"); - log.debug(String.format("%s sanitized to %s", folderName, sanitizedFolderName)); - return sanitizedFolderName; - } - - /** - * Will sanitize the image name and remove extension - * - * @param imageName path string - * @return Sanitized local path string - */ - public final String formatImageFileName(String imageName) { - return sanitize(imageName); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/PostService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/PostService.java deleted file mode 100644 index 66abf465..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/PostService.java +++ /dev/null @@ -1,73 +0,0 @@ -package tn.mnlr.vripper.services; - -import com.github.benmanes.caffeine.cache.Caffeine; -import com.github.benmanes.caffeine.cache.LoadingCache; -import lombok.extern.slf4j.Slf4j; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import reactor.core.Disposable; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; -import tn.mnlr.vripper.jpa.domain.Queued; -import tn.mnlr.vripper.services.domain.MultiPostScanParser; -import tn.mnlr.vripper.services.domain.MultiPostScanResult; -import tn.mnlr.vripper.tasks.AddPostRunnable; -import tn.mnlr.vripper.tasks.AddQueuedRunnable; - -import javax.annotation.PreDestroy; -import java.util.List; -import java.util.concurrent.ExecutionException; -import java.util.concurrent.TimeUnit; - -@Service -@Slf4j -public class PostService { - - private final DataService dataService; - private final ThreadPoolService threadPoolService; - private final LoadingCache cache; - private final Disposable disposable; - - @Autowired - public PostService( - DataService dataService, ThreadPoolService threadPoolService, EventBus eventBus) { - this.dataService = dataService; - this.threadPoolService = threadPoolService; - cache = - Caffeine.newBuilder() - .expireAfterWrite(5, TimeUnit.MINUTES) - .build(multiPostItem -> new MultiPostScanParser(multiPostItem).parse()); - disposable = - eventBus - .flux() - .filter(e -> e.getKind().equals(Event.Kind.SETTINGS_UPDATE)) - .subscribe(e -> this.cache.invalidateAll()); - } - - @PreDestroy - private void destroy() { - if (disposable != null) { - disposable.dispose(); - } - } - - public void processMultiPost(List queuedList) { - for (Queued queued : queuedList) { - if (queued.getPostId() != null) { - threadPoolService - .getGeneralExecutor() - .submit(new AddPostRunnable(queued.getPostId(), queued.getThreadId())); - } else { - threadPoolService.getGeneralExecutor().submit(new AddQueuedRunnable(queued)); - } - } - } - - public MultiPostScanResult get(Queued queued) throws ExecutionException { - return cache.get(queued); - } - - public void remove(String threadId) { - dataService.removeQueueLink(threadId); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/SettingsService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/SettingsService.java deleted file mode 100644 index 25a3c7e5..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/SettingsService.java +++ /dev/null @@ -1,305 +0,0 @@ -package tn.mnlr.vripper.services; - -import com.fasterxml.jackson.annotation.JsonProperty; -import com.fasterxml.jackson.core.type.TypeReference; -import com.fasterxml.jackson.databind.DeserializationFeature; -import com.fasterxml.jackson.databind.ObjectMapper; -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.extern.slf4j.Slf4j; -import org.apache.commons.codec.digest.DigestUtils; -import org.springframework.beans.factory.annotation.Value; -import org.springframework.core.io.Resource; -import org.springframework.stereotype.Service; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; -import tn.mnlr.vripper.exception.ValidationException; -import tn.mnlr.vripper.services.domain.Settings; - -import javax.annotation.PostConstruct; -import javax.annotation.PreDestroy; -import java.io.FileWriter; -import java.io.IOException; -import java.io.InputStream; -import java.nio.file.Files; -import java.nio.file.InvalidPathException; -import java.nio.file.Path; -import java.nio.file.Paths; -import java.util.ArrayList; -import java.util.HashSet; -import java.util.List; -import java.util.Set; - -import static java.nio.file.StandardOpenOption.*; - -@Service -@Setter -@Slf4j -public class SettingsService { - - private final Path configPath; - private final Path customProxiesPath; - private final ObjectMapper om = new ObjectMapper(); - - private final Set proxies = new HashSet<>(); - - private final EventBus eventBus; - - @Getter private Settings settings = new Settings(); - - @Value("classpath:proxies.json") - private Resource defaultProxies; - - public SettingsService( - @Value("${base.dir}") String baseDir, - @Value("${base.dir.name}") String baseDirName, - EventBus eventBus) { - this.eventBus = eventBus; - this.configPath = Paths.get(baseDir, baseDirName, "config.json"); - this.customProxiesPath = Paths.get(baseDir, baseDirName, "proxies.json"); - om.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); - } - - @PostConstruct - private void init() { - loadViperProxies(); - restore(); - eventBus.publishEvent(Event.wrap(Event.Kind.SETTINGS_UPDATE, settings)); - } - - private void loadViperProxies() { - try (InputStream defaultProxiesIs = defaultProxies.getInputStream()) { - List defaultProxies = om.readValue(defaultProxiesIs, new TypeReference<>() {}); - List customProxies = new ArrayList<>(); - if (customProxiesPath.toFile().exists() && customProxiesPath.toFile().isFile()) { - customProxies = om.readValue(customProxiesPath.toFile(), new TypeReference<>() {}); - } else { - if (!customProxiesPath.toFile().createNewFile()) { - log.warn("Unable to create " + customProxiesPath.toFile().getAbsolutePath()); - } else { - try (FileWriter fw = new FileWriter(customProxiesPath.toFile())) { - fw.append("[]").append(System.lineSeparator()); - fw.flush(); - } catch (IOException e) { - log.warn("Unable to create " + customProxiesPath.toFile().getAbsolutePath(), e); - } - } - } - proxies.addAll(customProxies); - proxies.addAll(defaultProxies); - } catch (IOException e) { - log.error("Failed to load vipergirls proxies list", e); - } - proxies.add("https://vipergirls.to"); - } - - public List getProxies() { - return new ArrayList<>(proxies); - } - - public void newSettings(Settings settings) { - - if (settings.getVLogin() != null && settings.getVLogin()) { - if (!this.settings.getVPassword().equals(settings.getVPassword())) { - settings.setVPassword(DigestUtils.md5Hex(settings.getVPassword())); - } - } else { - settings.setVUsername(""); - settings.setVPassword(""); - settings.setVThanks(false); - settings.setVLogin(false); - } - this.settings = settings; - - save(); - eventBus.publishEvent(Event.wrap(Event.Kind.SETTINGS_UPDATE, settings)); - } - - public void restore() { - try { - if (configPath.toFile().exists()) { - settings = om.readValue(configPath.toFile(), Settings.class); - } - } catch (IOException e) { - log.error("Failed restore user settings", e); - settings = new Settings(); - } - - if (settings.getDownloadPath() == null) { - settings.setDownloadPath(System.getProperty("user.home")); - } - - if (settings.getMaxThreads() == null) { - settings.setMaxThreads(4); - } - - if (settings.getMaxTotalThreads() == null) { - settings.setMaxTotalThreads(0); - } - - if (settings.getAutoStart() == null) { - settings.setAutoStart(true); - } - - if (settings.getVLogin() == null) { - settings.setVLogin(false); - } - - if (settings.getVUsername() == null) { - settings.setVUsername(""); - } - - if (settings.getVPassword() == null) { - settings.setVPassword(""); - } - - if (settings.getVThanks() == null) { - settings.setVThanks(false); - } - - if (settings.getDesktopClipboard() == null) { - settings.setDesktopClipboard(false); - } - - if (settings.getForceOrder() == null) { - settings.setForceOrder(false); - } - - if (settings.getSubLocation() == null) { - settings.setSubLocation(false); - } - - if (settings.getThreadSubLocation() == null) { - settings.setThreadSubLocation(false); - } - - if (settings.getClearCompleted() == null) { - settings.setClearCompleted(false); - } - - if (settings.getDarkTheme() == null) { - settings.setDarkTheme(false); - } - - if (settings.getAppendPostId() == null) { - settings.setAppendPostId(true); - } - - if (settings.getLeaveThanksOnStart() == null) { - settings.setLeaveThanksOnStart(false); - } - - if (settings.getConnectionTimeout() == null) { - settings.setConnectionTimeout(30); - } - - if (settings.getMaxAttempts() == null) { - settings.setMaxAttempts(5); - } - - if (settings.getVProxy() == null || !proxies.contains(settings.getVProxy())) { - settings.setVProxy("https://vipergirls.to"); - } - - if (settings.getMaxEventLog() == null) { - settings.setMaxEventLog(1000); - } - - try { - check(this.settings); - } catch (ValidationException e) { - log.error( - String.format("Your settings are invalid, either remove %s, or fix it", configPath), e); - SpringContext.close(); - } - - save(); - } - - public void save() { - try { - Files.write( - configPath, om.writeValueAsBytes(settings), CREATE, WRITE, TRUNCATE_EXISTING, SYNC); - } catch (IOException e) { - log.error("Failed to store user settings", e); - } - } - - @PreDestroy - private void destroy() { - save(); - } - - public void check(Settings settings) throws ValidationException { - - Path path; - try { - path = Paths.get(settings.getDownloadPath()); - } catch (InvalidPathException e) { - throw new ValidationException(String.format("%s is invalid", settings.getDownloadPath())); - } - if (!Files.exists(path)) { - throw new ValidationException(String.format("%s does not exist", settings.getDownloadPath())); - } else if (!Files.isDirectory(path)) { - throw new ValidationException( - String.format("%s is not a directory", settings.getDownloadPath())); - } - - if (settings.getMaxTotalThreads() < 0 || settings.getMaxTotalThreads() > 12) { - throw new ValidationException( - String.format( - "Invalid max global concurrent download settings, values must be in [%d,%d]", 0, 12)); - } - - if (settings.getMaxThreads() < 1 || settings.getMaxThreads() > 4) { - throw new ValidationException( - String.format( - "Invalid max concurrent download settings, values must be in [%d,%d]", 1, 4)); - } - - if (settings.getConnectionTimeout() < 1 || settings.getConnectionTimeout() > 300) { - throw new ValidationException( - String.format("Invalid connection timeout settings, values must be in [%d,%d]", 1, 300)); - } - - if (settings.getMaxAttempts() < 1 || settings.getMaxAttempts() > 10) { - throw new ValidationException( - String.format("Invalid maximum attempts settings, values must be in [%d,%d]", 1, 10)); - } - - if (settings.getMaxEventLog() < 100 || settings.getMaxEventLog() > 10_000) { - throw new ValidationException( - String.format( - "Invalid maximum event log record settings, values must be in [%d,%d]", 100, 10_000)); - } - - if (!settings.getVThanks() - && settings.getLeaveThanksOnStart() != null - && settings.getLeaveThanksOnStart()) { - throw new ValidationException("Invalid value, leave thanks must be checked"); - } - } - - public Theme getTheme() { - return new Theme(this.settings.getDarkTheme()); - } - - public void setTheme(Theme theme) { - this.settings.setDarkTheme(theme.darkTheme); - save(); - } - - @Getter - @NoArgsConstructor - public static class Theme { - - @JsonProperty("darkTheme") - private boolean darkTheme; - - Theme(boolean darkTheme) { - this.darkTheme = darkTheme; - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/ThreadPoolService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/ThreadPoolService.java deleted file mode 100644 index 90e341fc..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/ThreadPoolService.java +++ /dev/null @@ -1,20 +0,0 @@ -package tn.mnlr.vripper.services; - -import lombok.Getter; -import org.springframework.stereotype.Service; - -import java.util.concurrent.ExecutorService; -import java.util.concurrent.Executors; -import java.util.concurrent.TimeUnit; - -@Service -public class ThreadPoolService { - @Getter - private final ExecutorService generalExecutor = - Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors()); - - public void destroy() throws Exception { - generalExecutor.shutdown(); - generalExecutor.awaitTermination(5, TimeUnit.SECONDS); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/VGAuthService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/VGAuthService.java deleted file mode 100644 index b6ddc67d..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/VGAuthService.java +++ /dev/null @@ -1,159 +0,0 @@ -package tn.mnlr.vripper.services; - -import lombok.Getter; -import lombok.extern.slf4j.Slf4j; -import org.apache.http.NameValuePair; -import org.apache.http.client.entity.UrlEncodedFormEntity; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.client.protocol.HttpClientContext; -import org.apache.http.cookie.Cookie; -import org.apache.http.impl.client.BasicCookieStore; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.message.BasicNameValuePair; -import org.apache.http.util.EntityUtils; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.stereotype.Service; -import reactor.core.Disposable; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; -import tn.mnlr.vripper.exception.VripperException; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.tasks.LeaveThanksRunnable; - -import javax.annotation.PostConstruct; -import javax.annotation.PreDestroy; -import java.util.ArrayList; -import java.util.List; - -@Service -@Slf4j -public class VGAuthService { - - private final ConnectionService cm; - private final SettingsService settingsService; - private final ThreadPoolService threadPoolService; - - private final Disposable disposable; - - @Getter private final HttpClientContext context = HttpClientContext.create(); - @Getter private boolean authenticated = false; - @Getter private String loggedUser = ""; - - private final EventBus eventBus; - - @Autowired - public VGAuthService( - ConnectionService cm, - SettingsService settingsService, - ThreadPoolService threadPoolService, - EventBus eventBus) { - this.cm = cm; - this.settingsService = settingsService; - this.threadPoolService = threadPoolService; - this.eventBus = eventBus; - disposable = - eventBus - .flux() - .filter(p -> p.getKind().equals(Event.Kind.SETTINGS_UPDATE)) - .subscribe(e -> authenticate()); - } - - @PostConstruct - private void init() { - context.setCookieStore(new BasicCookieStore()); - authenticate(); - } - - @PreDestroy - private void destroy() { - disposable.dispose(); - } - - public void authenticate() { - - log.info("Authenticating using ViperGirls credentials"); - authenticated = false; - - if (!settingsService.getSettings().getVLogin()) { - log.debug("Authentication option is disabled"); - context.getCookieStore().clear(); - loggedUser = ""; - eventBus.publishEvent(Event.wrap(Event.Kind.VG_USER, loggedUser)); - return; - } - - String username = settingsService.getSettings().getVUsername(); - String password = settingsService.getSettings().getVPassword(); - - if (username == null || password == null || username.isEmpty() || password.isEmpty()) { - log.error("Cannot authenticate with ViperGirls credentials, username or password is empty"); - context.getCookieStore().clear(); - loggedUser = ""; - eventBus.publishEvent(Event.wrap(Event.Kind.VG_USER, loggedUser)); - return; - } - - HttpPost postAuth = - cm.buildHttpPost(settingsService.getSettings().getVProxy() + "/login.php?do=login", null); - List params = new ArrayList<>(); - params.add(new BasicNameValuePair("vb_login_username", username)); - - params.add(new BasicNameValuePair("cookieuser", "1")); - params.add(new BasicNameValuePair("do", "login")); - params.add(new BasicNameValuePair("vb_login_md5password", password)); - try { - postAuth.setEntity(new UrlEncodedFormEntity(params)); - } catch (Exception e) { - context.getCookieStore().clear(); - loggedUser = ""; - eventBus.publishEvent(Event.wrap(Event.Kind.VG_USER, loggedUser)); - log.error("Failed to authenticate user with " + settingsService.getSettings().getVProxy(), e); - return; - } - - postAuth.addHeader("Referer", settingsService.getSettings().getVProxy()); - postAuth.addHeader( - "Host", - settingsService.getSettings().getVProxy().replace("https://", "").replace("http://", "")); - - CloseableHttpClient client = cm.getClient().build(); - - try (CloseableHttpResponse response = client.execute(postAuth, context)) { - - if (response.getStatusLine().getStatusCode() / 100 != 2) { - throw new VripperException( - String.format( - "Unexpected response code returned %s", response.getStatusLine().getStatusCode())); - } - String responseBody = EntityUtils.toString(response.getEntity()); - log.debug(String.format("Authentication with ViperGirls response body:%n%s", responseBody)); - EntityUtils.consumeQuietly(response.getEntity()); - if (context.getCookieStore().getCookies().stream() - .map(Cookie::getName) - .noneMatch(e -> e.equals("vg_userid"))) { - log.error( - String.format( - "Failed to authenticate user with %s, missing vg_userid cookie", - settingsService.getSettings().getVProxy())); - return; - } - } catch (Exception e) { - context.getCookieStore().clear(); - loggedUser = ""; - eventBus.publishEvent(Event.wrap(Event.Kind.VG_USER, loggedUser)); - log.error("Failed to authenticate user with " + settingsService.getSettings().getVProxy(), e); - return; - } - authenticated = true; - loggedUser = username; - log.info(String.format("Authenticated: %s", username)); - eventBus.publishEvent(Event.wrap(Event.Kind.VG_USER, loggedUser)); - } - - public void leaveThanks(Post post) { - threadPoolService - .getGeneralExecutor() - .submit(new LeaveThanksRunnable(post, authenticated, context)); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/XpathService.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/XpathService.java deleted file mode 100644 index 8c045e81..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/XpathService.java +++ /dev/null @@ -1,32 +0,0 @@ -package tn.mnlr.vripper.services; - -import org.springframework.stereotype.Service; -import org.w3c.dom.Node; -import org.w3c.dom.NodeList; -import tn.mnlr.vripper.exception.XpathException; - -import javax.xml.xpath.XPath; -import javax.xml.xpath.XPathConstants; -import javax.xml.xpath.XPathFactory; - -@Service -public class XpathService { - - private final XPath xpath = XPathFactory.newInstance().newXPath(); - - public Node getAsNode(Node source, String xpathExpression) throws XpathException { - try { - return (Node) xpath.compile(xpathExpression).evaluate(source, XPathConstants.NODE); - } catch (Exception e) { - throw new XpathException(e); - } - } - - public NodeList getAsNodeList(Node source, String xpathExpression) throws XpathException { - try { - return (NodeList) xpath.compile(xpathExpression).evaluate(source, XPathConstants.NODESET); - } catch (Exception e) { - throw new XpathException(e); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/DownloadSpeed.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/DownloadSpeed.java deleted file mode 100644 index e8aa763d..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/DownloadSpeed.java +++ /dev/null @@ -1,40 +0,0 @@ -package tn.mnlr.vripper.services.domain; - -import lombok.Getter; - -import java.util.Objects; - -@Getter -public class DownloadSpeed { - - private final String speed; - - public DownloadSpeed(long bytes) { - speed = formatSI(bytes); - } - - private String formatSI(long bytes) { - return humanReadableByteCount(bytes, false); - } - - private String humanReadableByteCount(long bytes, boolean si) { - int unit = si ? 1000 : 1024; - if (bytes < unit) return bytes + " B"; - int exp = (int) (Math.log(bytes) / Math.log(unit)); - String pre = (si ? "kMGTPE" : "KMGTPE").charAt(exp - 1) + (si ? "" : "i"); - return String.format("%.1f %sB", bytes / Math.pow(unit, exp), pre); - } - - @Override - public boolean equals(Object o) { - if (this == o) return true; - if (o == null || getClass() != o.getClass()) return false; - DownloadSpeed that = (DownloadSpeed) o; - return Objects.equals(speed, that.speed); - } - - @Override - public int hashCode() { - return Objects.hash(speed); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/GlobalState.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/GlobalState.java deleted file mode 100644 index dbdbd1ad..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/GlobalState.java +++ /dev/null @@ -1,32 +0,0 @@ -package tn.mnlr.vripper.services.domain; - -import lombok.Getter; - -import java.util.Objects; - -@Getter -public class GlobalState { - - private final long running; - private final long remaining; - private final long error; - - public GlobalState(long running, long remaining, long error) { - this.running = running; - this.remaining = remaining; - this.error = error; - } - - @Override - public boolean equals(Object o) { - if (this == o) return true; - if (o == null || getClass() != o.getClass()) return false; - GlobalState that = (GlobalState) o; - return running == that.running && remaining == that.remaining && error == that.error; - } - - @Override - public int hashCode() { - return Objects.hash(running, remaining, error); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostItem.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostItem.java deleted file mode 100644 index eaebdd50..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostItem.java +++ /dev/null @@ -1,36 +0,0 @@ -package tn.mnlr.vripper.services.domain; - -import lombok.Getter; - -import java.util.List; - -@Getter -public class MultiPostItem { - private final String threadId; - private final String postId; - private final int number; - private final String title; - private final int imageCount; - private final String url; - private final List previews; - private final String hosts; - - public MultiPostItem( - String threadId, - String postId, - int number, - String title, - int imageCount, - String url, - List previews, - String hosts) { - this.threadId = threadId; - this.postId = postId; - this.number = number; - this.title = title; - this.imageCount = imageCount; - this.previews = previews; - this.url = url; - this.hosts = hosts; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostScanHandler.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostScanHandler.java deleted file mode 100644 index 61123460..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostScanHandler.java +++ /dev/null @@ -1,114 +0,0 @@ -package tn.mnlr.vripper.services.domain; - -import org.xml.sax.Attributes; -import org.xml.sax.helpers.DefaultHandler; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.host.Host; -import tn.mnlr.vripper.jpa.domain.Queued; -import tn.mnlr.vripper.services.SettingsService; - -import java.util.*; -import java.util.concurrent.atomic.AtomicInteger; -import java.util.stream.Collectors; - -public class MultiPostScanHandler extends DefaultHandler { - - private final Queued queued; - private final Collection supportedHosts; - private final SettingsService settingsService; - private final Map hostMap = new HashMap<>(); - private final List posts = new ArrayList<>(); - private String error; - private List previews = new ArrayList<>(); - private String threadTitle; - private String postId; - private String postTitle; - private int imageCount; - private int postCounter; - private int previewCounter = 0; - - public MultiPostScanHandler(Queued queued) { - this.queued = queued; - supportedHosts = SpringContext.getBeansOfType(Host.class).values(); - settingsService = SpringContext.getBean(SettingsService.class); - } - - public MultiPostScanResult getScanResult() { - return new MultiPostScanResult(posts, error); - } - - @Override - public void startDocument() {} - - @Override - public void startElement(String uri, String localName, String qName, Attributes attributes) { - - switch (qName.toLowerCase()) { - case "error": - error = attributes.getValue("details"); - break; - case "thread": - threadTitle = attributes.getValue("title").trim(); - break; - case "post": - imageCount = Integer.parseInt(attributes.getValue("imagecount").trim()); - postId = attributes.getValue("id").trim(); - postCounter = Integer.parseInt(attributes.getValue("number").trim()); - postTitle = - Optional.ofNullable(attributes.getValue("title")) - .map(e -> e.trim().isEmpty() ? null : e.trim()) - .orElse(threadTitle); - break; - case "image": - Optional.ofNullable(attributes.getValue("main_url")) - .map(String::trim) - .flatMap( - mainUrl -> - supportedHosts.stream().filter(host -> host.isSupported(mainUrl)).findFirst()) - .ifPresent( - host -> - Optional.ofNullable(hostMap.get(host)) - .ifPresentOrElse( - AtomicInteger::incrementAndGet, - () -> hostMap.put(host, new AtomicInteger(1)))); - if (previewCounter++ < 4) { - Optional.ofNullable(attributes.getValue("thumb_url")) - .map(String::trim) - .ifPresent(previews::add); - } - break; - } - } - - @Override - public void endElement(String uri, String localName, String qName) { - if ("post".equalsIgnoreCase(qName)) { - if (imageCount != 0) { - posts.add( - new MultiPostItem( - queued.getThreadId(), - postId, - postCounter, - postTitle, - imageCount, - String.format( - "%s/threads/?p=%s&viewfull=1#post%s", - settingsService.getSettings().getVProxy(), postId, postId), - previews, - hostMap.entrySet().stream() - .filter(v -> v.getValue().get() > 0) - .map(e -> e.getKey().getHost() + " (" + e.getValue().get() + ")") - .collect(Collectors.joining(", ")))); - queued.increment(); - } - previewCounter = 0; - previews = new ArrayList<>(); - hostMap.clear(); - } - } - - @Override - public void endDocument() { - queued.done(); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostScanParser.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostScanParser.java deleted file mode 100644 index 9cd1ba69..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostScanParser.java +++ /dev/null @@ -1,91 +0,0 @@ -package tn.mnlr.vripper.services.domain; - -import lombok.extern.slf4j.Slf4j; -import net.jodah.failsafe.Failsafe; -import org.apache.http.client.HttpClient; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.client.utils.URIBuilder; -import org.apache.http.util.EntityUtils; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.exception.DownloadException; -import tn.mnlr.vripper.exception.PostParseException; -import tn.mnlr.vripper.jpa.domain.Queued; -import tn.mnlr.vripper.services.ConnectionService; -import tn.mnlr.vripper.services.SettingsService; -import tn.mnlr.vripper.services.VGAuthService; - -import javax.xml.parsers.SAXParserFactory; -import java.io.BufferedInputStream; -import java.net.URISyntaxException; -import java.util.concurrent.atomic.AtomicReference; - -@Slf4j -public class MultiPostScanParser { - - private static final SAXParserFactory factory = SAXParserFactory.newInstance(); - private final Queued queued; - private final ConnectionService cm; - private final VGAuthService VGAuthService; - private final SettingsService settingsService; - - public MultiPostScanParser(Queued queued) { - this.queued = queued; - cm = SpringContext.getBean(ConnectionService.class); - VGAuthService = SpringContext.getBean(VGAuthService.class); - settingsService = SpringContext.getBean(SettingsService.class); - } - - public MultiPostScanResult parse() throws PostParseException { - - log.debug(String.format("Parsing thread %s", queued)); - HttpGet httpGet; - try { - URIBuilder uriBuilder = new URIBuilder(settingsService.getSettings().getVProxy() + "/vr.php"); - uriBuilder.setParameter("t", queued.getThreadId()); - httpGet = cm.buildHttpGet(uriBuilder.build(), null); - } catch (URISyntaxException e) { - throw new PostParseException(e); - } - - MultiPostScanHandler multiPostScanHandler = new MultiPostScanHandler(queued); - AtomicReference thr = new AtomicReference<>(); - log.debug(String.format("Requesting %s", httpGet)); - MultiPostScanResult multiPostScanResult = - Failsafe.with(cm.getRetryPolicy()) - .onFailure(e -> thr.set(e.getFailure())) - .get( - () -> { - HttpClient connection = cm.getClient().build(); - try (CloseableHttpResponse response = - (CloseableHttpResponse) - connection.execute(httpGet, VGAuthService.getContext())) { - if (response.getStatusLine().getStatusCode() / 100 != 2) { - throw new DownloadException( - String.format( - "Unexpected response code '%d' for %s", - response.getStatusLine().getStatusCode(), httpGet)); - } - - try { - factory - .newSAXParser() - .parse( - new BufferedInputStream(response.getEntity().getContent()), - multiPostScanHandler); - return multiPostScanHandler.getScanResult(); - } catch (Exception e) { - throw new PostParseException( - String.format("Failed to parse thread %s", queued), e); - } finally { - EntityUtils.consumeQuietly(response.getEntity()); - } - } - }); - if (thr.get() != null) { - log.error(String.format("parsing failed for thread %s", queued), thr.get()); - throw new PostParseException(thr.get()); - } - return multiPostScanResult; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostScanResult.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostScanResult.java deleted file mode 100644 index 127bf054..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/MultiPostScanResult.java +++ /dev/null @@ -1,16 +0,0 @@ -package tn.mnlr.vripper.services.domain; - -import lombok.Getter; - -import java.util.List; - -@Getter -public class MultiPostScanResult { - private final List posts; - private final String error; - - public MultiPostScanResult(List posts, String error) { - this.posts = posts; - this.error = error; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/PostScanHandler.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/PostScanHandler.java deleted file mode 100644 index c8d48b64..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/PostScanHandler.java +++ /dev/null @@ -1,116 +0,0 @@ -package tn.mnlr.vripper.services.domain; - -import lombok.extern.slf4j.Slf4j; -import org.xml.sax.Attributes; -import org.xml.sax.helpers.DefaultHandler; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.host.Host; -import tn.mnlr.vripper.jpa.domain.Image; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.services.SettingsService; - -import java.util.Collection; -import java.util.HashSet; -import java.util.Optional; -import java.util.Set; -import java.util.stream.Collectors; - -@Slf4j -class PostScanHandler extends DefaultHandler { - - private final Collection supportedHosts; - - private final String threadId; - private final String postId; - private final String postUrl; - private final Set images = new HashSet<>(); - private Set previews = new HashSet<>(); - private String threadTitle; - private String forum; - private String userHash; - private int index = 0; - - private Post parsedPost; - - PostScanHandler(String threadId, String postId) { - this.threadId = threadId; - this.postId = postId; - supportedHosts = SpringContext.getBeansOfType(Host.class).values(); - SettingsService settingsService = SpringContext.getBean(SettingsService.class); - postUrl = - String.format( - "%s/threads/%s/?p=%s&viewfull=1#post%s", - settingsService.getSettings().getVProxy(), this.threadId, this.postId, this.postId); - } - - public PostScanResult getParsedPost() { - return new PostScanResult(parsedPost, images); - } - - @Override - public void startDocument() {} - - @Override - public void startElement(String uri, String localName, String qName, Attributes attributes) { - - switch (qName.toLowerCase()) { - case "forum": - forum = attributes.getValue("title").trim(); - break; - case "user": - userHash = attributes.getValue("hash").trim(); - break; - case "thread": - threadTitle = attributes.getValue("title").trim(); - break; - case "post": - String postTitle = - Optional.ofNullable(attributes.getValue("title")) - .map(e -> e.trim().isEmpty() ? null : e.trim()) - .orElse(threadTitle); - parsedPost = new Post(postTitle, postUrl, postId, threadId, threadTitle, forum, userHash); - break; - case "image": - index++; - if (previews.size() < 4) { - Optional.ofNullable(attributes.getValue("thumb_url")) - .map(String::trim) - .ifPresent(previews::add); - } - - String mainUrl = - Optional.ofNullable(attributes.getValue("main_url")).map(String::trim).orElse(null); - if (mainUrl != null) { - Host foundHost = - supportedHosts.stream() - .filter(host -> host.isSupported(mainUrl)) - .findFirst() - .orElse(null); - if (foundHost != null) { - log.debug( - String.format( - "Found supported host %s for %s", - foundHost.getClass().getSimpleName(), mainUrl)); - images.add(new Image(postId, mainUrl, foundHost, index)); - } else { - log.warn(String.format("unsupported host for %s, skipping", mainUrl)); - } - } - break; - } - } - - @Override - public void endElement(String uri, String localName, String qName) { - if ("post".equalsIgnoreCase(qName)) { - parsedPost.setTotal(images.size()); - if (!previews.isEmpty()) { - parsedPost.setPreviews(previews); - } - parsedPost.setHosts( - images.stream().map(Image::getHost).map(Host::getHost).collect(Collectors.toSet())); - index = 0; - previews = new HashSet<>(); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/PostScanParser.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/PostScanParser.java deleted file mode 100644 index e484da05..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/PostScanParser.java +++ /dev/null @@ -1,86 +0,0 @@ -package tn.mnlr.vripper.services.domain; - -import lombok.extern.slf4j.Slf4j; -import net.jodah.failsafe.Failsafe; -import org.apache.http.client.HttpClient; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.client.utils.URIBuilder; -import org.apache.http.util.EntityUtils; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.exception.DownloadException; -import tn.mnlr.vripper.exception.PostParseException; -import tn.mnlr.vripper.services.ConnectionService; -import tn.mnlr.vripper.services.SettingsService; -import tn.mnlr.vripper.services.VGAuthService; - -import javax.xml.parsers.SAXParserFactory; -import java.net.URISyntaxException; -import java.util.concurrent.atomic.AtomicReference; - -@Slf4j -public class PostScanParser { - - private static final SAXParserFactory factory = SAXParserFactory.newInstance(); - - private final String threadId; - private final String postId; - private final ConnectionService cm; - private final VGAuthService VGAuthService; - private final SettingsService settingsService; - - public PostScanParser(String threadId, String postId) { - this.threadId = threadId; - this.postId = postId; - cm = SpringContext.getBean(ConnectionService.class); - VGAuthService = SpringContext.getBean(VGAuthService.class); - settingsService = SpringContext.getBean(SettingsService.class); - } - - public PostScanResult parse() throws PostParseException { - - log.debug(String.format("Parsing post %s", postId)); - HttpGet httpGet; - try { - URIBuilder uriBuilder = new URIBuilder(settingsService.getSettings().getVProxy() + "/vr.php"); - uriBuilder.setParameter("p", postId); - httpGet = cm.buildHttpGet(uriBuilder.build(), null); - } catch (URISyntaxException e) { - throw new PostParseException(e); - } - - AtomicReference thr = new AtomicReference<>(); - PostScanHandler postScanHandler = new PostScanHandler(threadId, postId); - log.debug(String.format("Requesting %s", httpGet)); - PostScanResult post = getPost(httpGet, postScanHandler, thr); - if (thr.get() != null) { - log.error( - String.format("parsing failed for thread %s, post %s", threadId, postId), thr.get()); - throw new PostParseException(thr.get()); - } - return post; - } - - private PostScanResult getPost( - HttpGet httpGet, PostScanHandler postScanHandler, AtomicReference thr) { - return Failsafe.with(cm.getRetryPolicy()) - .onFailure(e -> thr.set(e.getFailure())) - .get( - () -> { - HttpClient connection = cm.getClient().build(); - try (CloseableHttpResponse response = - (CloseableHttpResponse) connection.execute(httpGet, VGAuthService.getContext())) { - if (response.getStatusLine().getStatusCode() / 100 != 2) { - throw new DownloadException( - String.format( - "Unexpected response code '%d' for %s", - response.getStatusLine().getStatusCode(), httpGet)); - } - - factory.newSAXParser().parse(response.getEntity().getContent(), postScanHandler); - EntityUtils.consumeQuietly(response.getEntity()); - return postScanHandler.getParsedPost(); - } - }); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/PostScanResult.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/PostScanResult.java deleted file mode 100644 index 5ccb8b53..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/PostScanResult.java +++ /dev/null @@ -1,24 +0,0 @@ -package tn.mnlr.vripper.services.domain; - -import lombok.Getter; -import tn.mnlr.vripper.jpa.domain.Image; -import tn.mnlr.vripper.jpa.domain.Post; - -import java.util.Optional; -import java.util.Set; - -public class PostScanResult { - - private final Post post; - - @Getter private final Set images; - - public PostScanResult(Post post, Set images) { - this.post = post; - this.images = images; - } - - public Optional getPost() { - return Optional.ofNullable(post); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/Settings.java b/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/Settings.java deleted file mode 100644 index e2fa809b..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/services/domain/Settings.java +++ /dev/null @@ -1,108 +0,0 @@ -package tn.mnlr.vripper.services.domain; - -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonProperty; -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.extern.slf4j.Slf4j; - -@Getter -@Setter -@Slf4j -@NoArgsConstructor -@JsonInclude(JsonInclude.Include.NON_NULL) -public class Settings implements Cloneable { - - @JsonProperty("downloadPath") - private String downloadPath; - - @JsonProperty("maxThreads") - private Integer maxThreads; - - @JsonProperty("maxTotalThreads") - private Integer maxTotalThreads; - - @JsonProperty("autoStart") - private Boolean autoStart; - - @JsonProperty("vLogin") - private Boolean vLogin; - - @JsonProperty("vUsername") - private String vUsername; - - @JsonProperty("vPassword") - private String vPassword; - - @JsonProperty("vThanks") - private Boolean vThanks; - - @JsonProperty("desktopClipboard") - private Boolean desktopClipboard; - - @JsonProperty("forceOrder") - private Boolean forceOrder; - - @JsonProperty("subLocation") - private Boolean subLocation; - - @JsonProperty("threadSubLocation") - private Boolean threadSubLocation; - - @JsonProperty("clearCompleted") - private Boolean clearCompleted; - - @JsonProperty("darkTheme") - private Boolean darkTheme; - - @JsonProperty("appendPostId") - private Boolean appendPostId; - - @JsonProperty("leaveThanksOnStart") - private Boolean leaveThanksOnStart; - - @JsonProperty("connectionTimeout") - private Integer connectionTimeout; - - @JsonProperty("maxAttempts") - private Integer maxAttempts; - - @JsonProperty("vProxy") - private String vProxy; - - @JsonProperty("maxEventLog") - private Integer maxEventLog; - - @Override - public Object clone() { - Settings clone; - try { - clone = (Settings) super.clone(); - } catch (CloneNotSupportedException e) { - log.error(e.getMessage(), e); - clone = new Settings(); - } - clone.downloadPath = downloadPath; - clone.maxThreads = maxThreads; - clone.maxTotalThreads = maxTotalThreads; - clone.autoStart = autoStart; - clone.vLogin = vLogin; - clone.vUsername = vUsername; - clone.vPassword = vPassword; - clone.vThanks = vThanks; - clone.desktopClipboard = desktopClipboard; - clone.forceOrder = forceOrder; - clone.subLocation = subLocation; - clone.threadSubLocation = threadSubLocation; - clone.clearCompleted = clearCompleted; - clone.darkTheme = darkTheme; - clone.appendPostId = appendPostId; - clone.leaveThanksOnStart = leaveThanksOnStart; - clone.connectionTimeout = connectionTimeout; - clone.maxAttempts = maxAttempts; - clone.vProxy = vProxy; - clone.maxEventLog = maxEventLog; - return clone; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/tasks/AddPostRunnable.java b/vripper-server/src/main/java/tn/mnlr/vripper/tasks/AddPostRunnable.java deleted file mode 100644 index c838ed12..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/tasks/AddPostRunnable.java +++ /dev/null @@ -1,138 +0,0 @@ -package tn.mnlr.vripper.tasks; - -import lombok.extern.slf4j.Slf4j; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.Utils; -import tn.mnlr.vripper.download.DownloadService; -import tn.mnlr.vripper.exception.PostParseException; -import tn.mnlr.vripper.jpa.domain.Image; -import tn.mnlr.vripper.jpa.domain.LogEvent; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.jpa.domain.enums.Status; -import tn.mnlr.vripper.jpa.repositories.ILogEventRepository; -import tn.mnlr.vripper.services.DataService; -import tn.mnlr.vripper.services.MetadataService; -import tn.mnlr.vripper.services.SettingsService; -import tn.mnlr.vripper.services.VGAuthService; -import tn.mnlr.vripper.services.domain.PostScanParser; -import tn.mnlr.vripper.services.domain.PostScanResult; - -import java.time.LocalDateTime; -import java.util.Map; -import java.util.Set; - -import static tn.mnlr.vripper.jpa.domain.LogEvent.Status.ERROR; -import static tn.mnlr.vripper.jpa.domain.LogEvent.Status.PROCESSING; - -@Slf4j -public class AddPostRunnable implements Runnable { - - private final String postId; - private final String threadId; - private final DataService dataService; - private final MetadataService metadataService; - private final SettingsService settingsService; - private final VGAuthService VGAuthService; - private final LogEvent logEvent; - private final ILogEventRepository eventRepository; - private final String link; - private final DownloadService downloadService; - - public AddPostRunnable(String postId, String threadId) { - this.postId = postId; - this.threadId = threadId; - this.dataService = SpringContext.getBean(DataService.class); - this.metadataService = SpringContext.getBean(MetadataService.class); - this.settingsService = SpringContext.getBean(SettingsService.class); - this.downloadService = SpringContext.getBean(DownloadService.class); - this.VGAuthService = SpringContext.getBean(VGAuthService.class); - this.eventRepository = SpringContext.getBean(ILogEventRepository.class); - link = - settingsService.getSettings().getVProxy() - + String.format("/threads/%s?%s", threadId, (postId != null ? "p=" + postId : "")); - logEvent = - new LogEvent( - LogEvent.Type.POST, - LogEvent.Status.PENDING, - LocalDateTime.now(), - String.format("Processing %s", link)); - eventRepository.save(logEvent); - } - - @Override - public void run() { - - try { - logEvent.setStatus(PROCESSING); - eventRepository.update(logEvent); - if (dataService.exists(postId)) { - log.warn(String.format("skipping %s, already loaded", postId)); - logEvent.setMessage(String.format("Gallery %s is already loaded", link)); - logEvent.setStatus(ERROR); - eventRepository.update(logEvent); - return; - } - - PostScanParser postScanParser = new PostScanParser(threadId, postId); - - PostScanResult postScanResult; - try { - postScanResult = postScanParser.parse(); - } catch (PostParseException e) { - String error = String.format("parsing failed for gallery %s", link); - log.error(error, e); - logEvent.setMessage(error + "\n" + Utils.throwableToString(e)); - logEvent.setStatus(ERROR); - eventRepository.update(logEvent); - return; - } - if (postScanResult.getPost().isEmpty()) { - String error = String.format("Gallery %s contains no galleries", link); - log.error(error); - logEvent.setMessage(error); - logEvent.setStatus(ERROR); - eventRepository.update(logEvent); - return; - } - if (postScanResult.getImages().isEmpty()) { - String error = String.format("Gallery %s contains no images to download", link); - log.error(error); - logEvent.setMessage(error); - logEvent.setStatus(ERROR); - eventRepository.update(logEvent); - return; - } - - Post post = postScanResult.getPost().get(); - Set images = postScanResult.getImages(); - - dataService.newPost(post, images); - metadataService.startFetchingMetadata(post); - - if (settingsService.getSettings().getAutoStart()) { - log.debug("Auto start downloads option is enabled"); - post.setStatus(Status.PENDING); - downloadService.enqueue(Map.of(post, images)); - log.debug(String.format("Done enqueuing jobs for %s", post.getUrl())); - } else { - post.setStatus(Status.STOPPED); - log.debug("Auto start downloads option is disabled"); - } - if (settingsService.getSettings().getLeaveThanksOnStart() != null - && !settingsService.getSettings().getLeaveThanksOnStart()) { - VGAuthService.leaveThanks(post); - } - dataService.updatePostStatus(post.getStatus(), post.getId()); - logEvent.setMessage( - String.format("Gallery %s is successfully added to download queue", link)); - logEvent.setStatus(LogEvent.Status.DONE); - eventRepository.update(logEvent); - } catch (Exception e) { - String error = String.format("Error when adding gallery %s", link); - log.error(error, e); - logEvent.setMessage(error + "\n" + Utils.throwableToString(e)); - logEvent.setStatus(ERROR); - eventRepository.update(logEvent); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/tasks/AddQueuedRunnable.java b/vripper-server/src/main/java/tn/mnlr/vripper/tasks/AddQueuedRunnable.java deleted file mode 100644 index daff6e5a..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/tasks/AddQueuedRunnable.java +++ /dev/null @@ -1,103 +0,0 @@ -package tn.mnlr.vripper.tasks; - -import lombok.extern.slf4j.Slf4j; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.Utils; -import tn.mnlr.vripper.jpa.domain.LogEvent; -import tn.mnlr.vripper.jpa.domain.Queued; -import tn.mnlr.vripper.jpa.repositories.ILogEventRepository; -import tn.mnlr.vripper.services.DataService; -import tn.mnlr.vripper.services.PostService; -import tn.mnlr.vripper.services.ThreadPoolService; -import tn.mnlr.vripper.services.domain.MultiPostScanResult; - -import java.time.LocalDateTime; - -import static tn.mnlr.vripper.jpa.domain.LogEvent.Status.ERROR; -import static tn.mnlr.vripper.jpa.domain.LogEvent.Status.PROCESSING; - -@Slf4j -public class AddQueuedRunnable implements Runnable { - - private final Queued queued; - private final ThreadPoolService threadPoolService; - private final DataService dataService; - private final ILogEventRepository eventRepository; - private final PostService postService; - private final LogEvent logEvent; - - public AddQueuedRunnable(Queued queued) { - this.queued = queued; - threadPoolService = SpringContext.getBean(ThreadPoolService.class); - dataService = SpringContext.getBean(DataService.class); - eventRepository = SpringContext.getBean(ILogEventRepository.class); - postService = SpringContext.getBean(PostService.class); - logEvent = - new LogEvent( - LogEvent.Type.QUEUED, - LogEvent.Status.PENDING, - LocalDateTime.now(), - String.format("Processing multi-post link %s", queued.getLink())); - eventRepository.save(logEvent); - } - - @Override - public void run() { - try { - logEvent.setStatus(PROCESSING); - eventRepository.update(logEvent); - MultiPostScanResult multiPostScanResult = postService.get(queued); - if (multiPostScanResult == null) { - String message = String.format("Fetching multi-post link %s failed", queued.getLink()); - logEvent.setStatus(ERROR); - logEvent.setMessage(message); - eventRepository.update(logEvent); - return; - } else if (multiPostScanResult.getError() != null) { - String message = - "Nothing found for " + queued.getLink() + "\n" + multiPostScanResult.getError(); - logEvent.setStatus(ERROR); - logEvent.setMessage(message); - eventRepository.update(logEvent); - return; - } else if (multiPostScanResult.getPosts().isEmpty()) { - String message = "Nothing found for " + queued.getLink(); - logEvent.setStatus(ERROR); - logEvent.setMessage(message); - eventRepository.update(logEvent); - return; - } - queued.setTotal(multiPostScanResult.getPosts().size()); - queued.done(); - if (multiPostScanResult.getPosts().size() == 1) { - threadPoolService - .getGeneralExecutor() - .submit( - new AddPostRunnable( - multiPostScanResult.getPosts().get(0).getPostId(), - multiPostScanResult.getPosts().get(0).getThreadId())); - logEvent.setStatus(LogEvent.Status.DONE); - logEvent.setMessage(String.format("Link %s is added to download queue", queued.getLink())); - } else { - if (dataService.findQueuedByThreadId(queued.getThreadId()).isEmpty()) { - dataService.newQueueLink(queued); - logEvent.setStatus(LogEvent.Status.DONE); - logEvent.setMessage( - String.format("Link %s is added to multi-post links", queued.getLink())); - } else { - log.info(String.format("Link %s is already loaded", queued.getLink())); - logEvent.setStatus(LogEvent.Status.ERROR); - logEvent.setMessage( - String.format("%s has already been added to multi-post links", queued.getLink())); - } - } - eventRepository.update(logEvent); - } catch (Exception e) { - String error = String.format("Error when adding multi-post link %s", queued.getLink()); - log.error(error, e); - logEvent.setMessage(error + "\n" + Utils.throwableToString(e)); - logEvent.setStatus(ERROR); - eventRepository.update(logEvent); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/tasks/LeaveThanksRunnable.java b/vripper-server/src/main/java/tn/mnlr/vripper/tasks/LeaveThanksRunnable.java deleted file mode 100644 index 9e8515da..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/tasks/LeaveThanksRunnable.java +++ /dev/null @@ -1,134 +0,0 @@ -package tn.mnlr.vripper.tasks; - -import lombok.extern.slf4j.Slf4j; -import org.apache.http.NameValuePair; -import org.apache.http.client.entity.UrlEncodedFormEntity; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.client.protocol.HttpClientContext; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.message.BasicNameValuePair; -import org.apache.http.util.EntityUtils; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.Utils; -import tn.mnlr.vripper.jpa.domain.LogEvent; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.jpa.repositories.ILogEventRepository; -import tn.mnlr.vripper.services.ConnectionService; -import tn.mnlr.vripper.services.DataService; -import tn.mnlr.vripper.services.SettingsService; - -import java.io.UnsupportedEncodingException; -import java.time.LocalDateTime; -import java.util.ArrayList; -import java.util.List; - -@Slf4j -public class LeaveThanksRunnable implements Runnable { - - private final ConnectionService cm; - private final DataService dataService; - private final HttpClientContext context; - private final SettingsService settingsService; - private final ILogEventRepository eventRepository; - private final boolean authenticated; - private final Post post; - private final LogEvent logEvent; - - public LeaveThanksRunnable(Post post, boolean authenticated, HttpClientContext context) { - this.post = post; - this.authenticated = authenticated; - this.context = context; - cm = SpringContext.getBean(ConnectionService.class); - dataService = SpringContext.getBean(DataService.class); - eventRepository = SpringContext.getBean(ILogEventRepository.class); - settingsService = SpringContext.getBean(SettingsService.class); - logEvent = - new LogEvent( - LogEvent.Type.THANKS, - LogEvent.Status.PENDING, - LocalDateTime.now(), - String.format("Leaving thanks for %s", post.getUrl())); - eventRepository.save(logEvent); - } - - @Override - public void run() { - try { - logEvent.setStatus(LogEvent.Status.PROCESSING); - eventRepository.update(logEvent); - if (!settingsService.getSettings().getVLogin()) { - logEvent.setMessage( - String.format( - "Will not send a like for %s\nAuthentication with ViperGirls option is disabled", - post.getUrl())); - logEvent.setStatus(LogEvent.Status.DONE); - eventRepository.update(logEvent); - return; - } - if (!settingsService.getSettings().getVThanks()) { - logEvent.setMessage( - String.format( - "Will not send a like for %s\nLeave thanks option is disabled", post.getUrl())); - logEvent.setStatus(LogEvent.Status.DONE); - eventRepository.update(logEvent); - return; - } - if (!authenticated) { - logEvent.setMessage( - String.format("Will not send a like for %s\nYou are not authenticated", post.getUrl())); - logEvent.setStatus(LogEvent.Status.ERROR); - eventRepository.update(logEvent); - return; - } - if (post.isThanked()) { - logEvent.setMessage( - String.format("Will not send a like for %s\nAlready left a thanks", post.getUrl())); - logEvent.setStatus(LogEvent.Status.DONE); - eventRepository.update(logEvent); - return; - } - - HttpPost postThanks = - cm.buildHttpPost(settingsService.getSettings().getVProxy() + "/post_thanks.php", null); - List params = new ArrayList<>(); - params.add(new BasicNameValuePair("do", "post_thanks_add")); - params.add(new BasicNameValuePair("using_ajax", "1")); - params.add(new BasicNameValuePair("p", post.getPostId())); - params.add(new BasicNameValuePair("securitytoken", post.getSecurityToken())); - try { - postThanks.setEntity(new UrlEncodedFormEntity(params)); - } catch (UnsupportedEncodingException e) { - String error = String.format("Request error for %s", post.getUrl()); - log.error(error, e); - logEvent.setMessage(error + "\n" + Utils.throwableToString(e)); - logEvent.setStatus(LogEvent.Status.ERROR); - eventRepository.update(logEvent); - return; - } - - postThanks.addHeader("Referer", settingsService.getSettings().getVProxy()); - postThanks.addHeader( - "Host", - settingsService.getSettings().getVProxy().replace("https://", "").replace("http://", "")); - - CloseableHttpClient client = cm.getClient().build(); - - try (CloseableHttpResponse response = client.execute(postThanks, context)) { - if (response.getStatusLine().getStatusCode() / 100 == 2) { - post.setThanked(true); - dataService.updatePostThanks(post.isThanked(), post.getId()); - } - EntityUtils.consumeQuietly(response.getEntity()); - } - logEvent.setStatus(LogEvent.Status.DONE); - eventRepository.update(logEvent); - } catch (Exception e) { - String error = String.format("Failed to leave a thanks for %s", post.getUrl()); - log.error(error, e); - logEvent.setMessage(error + "\n" + Utils.throwableToString(e)); - logEvent.setStatus(LogEvent.Status.ERROR); - eventRepository.update(logEvent); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/tasks/LinkScanRunnable.java b/vripper-server/src/main/java/tn/mnlr/vripper/tasks/LinkScanRunnable.java deleted file mode 100644 index 8b14f31f..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/tasks/LinkScanRunnable.java +++ /dev/null @@ -1,109 +0,0 @@ -package tn.mnlr.vripper.tasks; - -import lombok.extern.slf4j.Slf4j; -import org.springframework.lang.NonNull; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.Utils; -import tn.mnlr.vripper.jpa.domain.LogEvent; -import tn.mnlr.vripper.jpa.domain.Queued; -import tn.mnlr.vripper.jpa.repositories.impl.LogEventRepository; -import tn.mnlr.vripper.services.PostService; -import tn.mnlr.vripper.services.SettingsService; - -import java.time.LocalDateTime; -import java.util.ArrayList; -import java.util.List; -import java.util.regex.Matcher; -import java.util.regex.Pattern; - -import static tn.mnlr.vripper.jpa.domain.LogEvent.Status.ERROR; - -@Slf4j -public class LinkScanRunnable implements Runnable { - - private static final Object LOCK = new Object(); - private final List urlList; - private final SettingsService settingsService; - private final PostService postService; - private final LogEventRepository eventRepository; - private final LogEvent logEvent; - - public LinkScanRunnable(@NonNull List urlList) { - this.urlList = urlList; - settingsService = SpringContext.getBean(SettingsService.class); - postService = SpringContext.getBean(PostService.class); - eventRepository = SpringContext.getBean(LogEventRepository.class); - logEvent = - new LogEvent( - LogEvent.Type.SCAN, - LogEvent.Status.PENDING, - LocalDateTime.now(), - "Links to scan:\n\t" + String.join("\n\t", urlList)); - eventRepository.save(logEvent); - } - - @Override - public void run() { - synchronized (LOCK) { - try { - logEvent.setStatus(LogEvent.Status.PROCESSING); - eventRepository.update(logEvent); - ArrayList queuedList = new ArrayList<>(); - List unsupported = new ArrayList<>(); - List unrecognized = new ArrayList<>(); - for (String url : urlList) { - log.debug(String.format("Starting to process thread: %s", url)); - if (!url.startsWith(settingsService.getSettings().getVProxy())) { - log.error(String.format("Unsupported link %s", url)); - unsupported.add(url); - continue; - } - - String threadId, postId; - Matcher m = - Pattern.compile( - Pattern.quote(settingsService.getSettings().getVProxy()) - + "/threads/(\\d+)((.*p=)(\\d+))?") - .matcher(url); - if (m.find()) { - threadId = m.group(1); - postId = m.group(4); - } else { - log.error(String.format("Cannot retrieve thread id from URL %s", url)); - unrecognized.add(url); - continue; - } - queuedList.add(new Queued(url, threadId, postId)); - } - StringBuilder errorMessage = new StringBuilder(); - if (!unsupported.isEmpty()) { - errorMessage - .append("Unsupported links:\n\t") - .append(String.join("\n\t", unsupported)) - .append("\n\n"); - } - if (!unrecognized.isEmpty()) { - errorMessage - .append("Unrecognized links:\n\t") - .append(String.join("\n\t", unrecognized)) - .append("\n\n"); - } - - postService.processMultiPost(queuedList); - if (!unsupported.isEmpty() || !unrecognized.isEmpty()) { - logEvent.setStatus(LogEvent.Status.ERROR); - logEvent.setMessage("Some links failed to be scanned: \n" + errorMessage); - } else { - logEvent.setStatus(LogEvent.Status.DONE); - } - eventRepository.update(logEvent); - } catch (Exception e) { - String error = "Error when scanning links"; - log.error(error, e); - logEvent.setMessage(error + "\n" + Utils.throwableToString(e)); - logEvent.setStatus(ERROR); - eventRepository.update(logEvent); - } - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/tasks/MetadataRunnable.java b/vripper-server/src/main/java/tn/mnlr/vripper/tasks/MetadataRunnable.java deleted file mode 100644 index 7067128b..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/tasks/MetadataRunnable.java +++ /dev/null @@ -1,246 +0,0 @@ -package tn.mnlr.vripper.tasks; - -import lombok.Getter; -import lombok.NonNull; -import lombok.extern.slf4j.Slf4j; -import net.jodah.failsafe.Failsafe; -import org.apache.http.client.HttpClient; -import org.apache.http.client.methods.AbstractExecutionAwareRequest; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.client.protocol.HttpClientContext; -import org.apache.http.util.EntityUtils; -import org.w3c.dom.Document; -import org.w3c.dom.Node; -import tn.mnlr.vripper.SpringContext; -import tn.mnlr.vripper.Utils; -import tn.mnlr.vripper.download.DownloadJob; -import tn.mnlr.vripper.exception.DownloadException; -import tn.mnlr.vripper.exception.PostParseException; -import tn.mnlr.vripper.jpa.domain.LogEvent; -import tn.mnlr.vripper.jpa.domain.Metadata; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.jpa.repositories.ILogEventRepository; -import tn.mnlr.vripper.services.*; - -import java.time.LocalDateTime; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.List; -import java.util.concurrent.atomic.AtomicBoolean; -import java.util.concurrent.atomic.AtomicReference; -import java.util.stream.Collectors; - -import static tn.mnlr.vripper.jpa.domain.LogEvent.Status.ERROR; - -@Slf4j -public class MetadataRunnable implements Runnable { - - private static final List dictionary = - Arrays.asList("download", "link", "rapidgator", "filefactory", "filefox"); - - @Getter private final Post post; - - private final DataService dataService; - private final ILogEventRepository eventRepository; - - private final LogEvent logEvent; - private final ConnectionService cm; - private final HtmlProcessorService htmlProcessorService; - private final XpathService xpathService; - - private final HttpClientContext context = HttpClientContext.create(); - private final MetadataService metadataService; - private volatile boolean stopped = false; - @Getter private volatile boolean finished = false; - - public MetadataRunnable(@NonNull Post post) { - this.post = post; - dataService = SpringContext.getBean(DataService.class); - eventRepository = SpringContext.getBean(ILogEventRepository.class); - cm = SpringContext.getBean(ConnectionService.class); - VGAuthService vgAuthService = SpringContext.getBean(VGAuthService.class); - htmlProcessorService = SpringContext.getBean(HtmlProcessorService.class); - xpathService = SpringContext.getBean(XpathService.class); - metadataService = SpringContext.getBean(MetadataService.class); - - context.setCookieStore(vgAuthService.getContext().getCookieStore()); - context.setAttribute( - DownloadJob.ContextAttributes.OPEN_CONNECTION.toString(), - Collections.synchronizedList(new ArrayList())); - - logEvent = - new LogEvent( - LogEvent.Type.METADATA, - LogEvent.Status.PENDING, - LocalDateTime.now(), - "Fetching metadata for " + post.getUrl()); - eventRepository.save(logEvent); - } - - @Override - public void run() { - - try { - logEvent.setStatus(LogEvent.Status.PROCESSING); - eventRepository.update(logEvent); - Metadata metadata = fetchMetadata(post.getPostId(), post.getThreadId(), post.getUrl()); - if (metadata != null && !stopped) { - dataService.setMetadata(post, metadata); - logEvent.setStatus(LogEvent.Status.DONE); - } else { - logEvent.setStatus(ERROR); - logEvent.setMessage(String.format("Fetching metadata for %s failed", post.getUrl())); - } - eventRepository.update(logEvent); - } catch (Exception e) { - String message = String.format("Failed to fetch metadata for %s", post.getUrl()); - log.error(message, e); - logEvent.setMessage(message + "\n" + Utils.throwableToString(e)); - logEvent.setStatus(ERROR); - eventRepository.update(logEvent); - } finally { - if (stopped) { - String message = String.format("Fetching metadata for %s interrupted", post.getUrl()); - logEvent.setStatus(LogEvent.Status.DONE); - logEvent.setMessage(message); - eventRepository.update(logEvent); - } - finished = true; - metadataService.stopFetchingMetadata(List.of(post.getPostId())); - } - } - - private Metadata fetchMetadata(String postId, String threadId, String url) { - HttpGet httpGet = cm.buildHttpGet(url, context); - AtomicReference metadataReference = new AtomicReference<>(); - Failsafe.with(cm.getRetryPolicy()) - .onFailure( - e -> - log.error( - String.format("Error occurred when getting post metadata, postId %s", postId), - e.getFailure())) - .run( - () -> { - if (stopped) { - return; - } - HttpClient connection = cm.getClient().build(); - try (CloseableHttpResponse response = - (CloseableHttpResponse) connection.execute(httpGet, context)) { - if (stopped) { - return; - } - if (response.getStatusLine().getStatusCode() / 100 != 2) { - throw new DownloadException( - String.format( - "Unexpected response code '%d' for %s", - response.getStatusLine().getStatusCode(), httpGet)); - } - try { - Document document = - htmlProcessorService.clean(EntityUtils.toString(response.getEntity())); - - if (stopped) { - return; - } - - Node postNode = - xpathService.getAsNode( - document, - String.format( - "//li[@id='post_%s']/div[contains(@class, 'postdetails')]", postId)); - - if (stopped) { - return; - } - - String postedBy = - xpathService - .getAsNode( - postNode, - "./div[contains(@class, 'userinfo')]//a[contains(@class, 'username')]//font") - .getTextContent() - .trim(); - - if (stopped) { - return; - } - - Metadata metadata = new Metadata(); - metadata.setPostedBy(postedBy); - - if (stopped) { - return; - } - - Node node = - xpathService.getAsNode( - document, String.format("//div[@id='post_message_%s']", postId)); - metadata.setResolvedNames(findTitleInContent(node)); - metadata.setPostId(postId); - - if (stopped) { - return; - } - metadataReference.set(metadata); - } catch (Exception e) { - if (stopped) { - log.warn(e.getMessage(), e); - return; - } - throw new PostParseException( - String.format("Failed to parse thread %s, post %s", threadId, postId), e); - } finally { - EntityUtils.consumeQuietly(response.getEntity()); - } - } - }); - return metadataReference.get(); - } - - private List findTitleInContent(Node node) { - List altTitle = new ArrayList<>(); - findTitle(node, altTitle, new AtomicBoolean(true)); - return altTitle.stream().distinct().collect(Collectors.toList()); - } - - private void findTitle(Node node, List altTitle, AtomicBoolean keepGoing) { - if (!keepGoing.get()) { - return; - } - if (node.getNodeName().equals("a") || node.getNodeName().equals("img")) { - keepGoing.set(false); - return; - } - if (node.getNodeType() == Node.ELEMENT_NODE) { - for (int i = 0; i < node.getChildNodes().getLength(); i++) { - Node item = node.getChildNodes().item(i); - findTitle(item, altTitle, keepGoing); - if (!keepGoing.get()) { - return; - } - } - - } else if (node.getNodeType() == Node.TEXT_NODE) { - String text = node.getTextContent().trim(); - if (!text.isBlank() - && dictionary.stream().noneMatch(e -> text.toLowerCase().contains(e.toLowerCase()))) { - altTitle.add(text); - } - } - } - - public void stop() { - this.stopped = true; - List requests = - (List) - this.context.getAttribute(DownloadJob.ContextAttributes.OPEN_CONNECTION.toString()); - if (requests != null) { - for (AbstractExecutionAwareRequest request : requests) { - request.abort(); - } - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/EventLogRestEndpoint.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/EventLogRestEndpoint.java deleted file mode 100644 index 41531758..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/EventLogRestEndpoint.java +++ /dev/null @@ -1,27 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints; - -import lombok.extern.slf4j.Slf4j; -import org.springframework.http.HttpStatus; -import org.springframework.web.bind.annotation.CrossOrigin; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.ResponseStatus; -import org.springframework.web.bind.annotation.RestController; -import tn.mnlr.vripper.jpa.repositories.ILogEventRepository; - -@Slf4j -@RestController -@CrossOrigin(value = "*") -public class EventLogRestEndpoint { - - private final ILogEventRepository eventRepository; - - public EventLogRestEndpoint(ILogEventRepository eventRepository) { - this.eventRepository = eventRepository; - } - - @GetMapping("/events/clear") - @ResponseStatus(value = HttpStatus.OK) - public void clear() { - eventRepository.deleteAll(); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/PostRestEndpoint.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/PostRestEndpoint.java deleted file mode 100644 index ad14ea29..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/PostRestEndpoint.java +++ /dev/null @@ -1,268 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints; - -import lombok.NonNull; -import lombok.extern.slf4j.Slf4j; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.http.HttpStatus; -import org.springframework.web.bind.annotation.*; -import tn.mnlr.vripper.download.DownloadService; -import tn.mnlr.vripper.jpa.domain.Metadata; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.jpa.domain.Queued; -import tn.mnlr.vripper.services.*; -import tn.mnlr.vripper.services.domain.MultiPostItem; -import tn.mnlr.vripper.services.domain.MultiPostScanResult; -import tn.mnlr.vripper.tasks.AddPostRunnable; -import tn.mnlr.vripper.tasks.LinkScanRunnable; -import tn.mnlr.vripper.web.restendpoints.domain.*; -import tn.mnlr.vripper.web.restendpoints.exceptions.BadRequestException; -import tn.mnlr.vripper.web.restendpoints.exceptions.NotFoundException; -import tn.mnlr.vripper.web.restendpoints.exceptions.ServerErrorException; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.Optional; -import java.util.stream.Collectors; - -@Slf4j -@RestController -@CrossOrigin(value = "*") -public class PostRestEndpoint { - - private static final Object LOCK = new Object(); - private final DataService dataService; - private final PathService pathService; - private final DownloadService downloadService; - private final PostService postService; - private final ThreadPoolService threadPoolService; - private final SettingsService settingsService; - - @Autowired - public PostRestEndpoint( - DataService dataService, - PathService pathService, - DownloadService downloadService, - PostService postService, - ThreadPoolService threadPoolService, - SettingsService settingsService) { - this.dataService = dataService; - this.pathService = pathService; - this.downloadService = downloadService; - this.postService = postService; - this.threadPoolService = threadPoolService; - this.settingsService = settingsService; - } - - @PostMapping("/post") - @ResponseStatus(code = HttpStatus.OK) - public void processPost(@RequestBody ThreadUrl _url) { - synchronized (LOCK) { - if (_url.getUrl() == null || _url.getUrl().isBlank()) { - log.error("Cannot process empty requests"); - throw new BadRequestException("Cannot process empty requests"); - } - List urlList = - Arrays.stream(_url.getUrl().split("\\r?\\n")) - .map(String::trim) - .filter(e -> !e.isEmpty()) - .collect(Collectors.toList()); - threadPoolService.getGeneralExecutor().submit(new LinkScanRunnable(urlList)); - } - } - - @PostMapping("/post/restart") - @ResponseStatus(value = HttpStatus.OK) - public void restartPost(@RequestBody @NonNull List postIds) { - synchronized (LOCK) { - downloadService.restartAll( - postIds.stream().map(PostId::getPostId).collect(Collectors.toList())); - } - } - - @PostMapping("/post/add") - @ResponseStatus(value = HttpStatus.OK) - public void addPost(@RequestBody List posts) { - synchronized (LOCK) { - for (PostToAdd post : posts) { - threadPoolService - .getGeneralExecutor() - .submit(new AddPostRunnable(post.getPostId(), post.getThreadId())); - } - } - } - - @GetMapping("/post/path/{postId}") - @ResponseStatus(value = HttpStatus.OK) - public DownloadPath folderPath(@PathVariable("postId") String postId) { - synchronized (LOCK) { - return getDownloadPath(postId); - } - } - - private DownloadPath getDownloadPath(String postId) { - Optional _post = dataService.findPostByPostId(postId); - if (_post.isPresent()) { - Post post = _post.get(); - if (post.getDownloadDirectory() == null) { - log.error("Download has not been started yet for this post"); - throw new NotFoundException("Download has not been started yet for this post"); - } else { - return new DownloadPath( - pathService.calcDownloadDirectory(post, settingsService.getSettings()).getPath()); - } - } else { - log.error(String.format("Unable to find post with postId = %s", postId)); - throw new NotFoundException(String.format("Unable to find post with postId = %s", postId)); - } - } - - @PostMapping("/post/restart/all") - @ResponseStatus(value = HttpStatus.OK) - public void restartPost() { - synchronized (LOCK) { - downloadService.restartAll(null); - } - } - - @PostMapping("/post/stop") - @ResponseStatus(value = HttpStatus.OK) - public void stop(@RequestBody @NonNull List postIds) { - synchronized (LOCK) { - downloadService.stopAll(postIds.stream().map(PostId::getPostId).collect(Collectors.toList())); - } - } - - @PostMapping("/post/stop/all") - @ResponseStatus(value = HttpStatus.OK) - public void stopAll() { - synchronized (LOCK) { - downloadService.stopAll(null); - } - } - - @PostMapping("/post/remove") - @ResponseStatus(value = HttpStatus.OK) - public List remove(@RequestBody @NonNull List postIds) { - synchronized (LOCK) { - List result = new ArrayList<>(); - List collect = - postIds.stream() - .map(PostId::getPostId) - .peek(e -> result.add(new RemoveResult(e))) - .collect(Collectors.toList()); - downloadService.stopAll(collect); - dataService.removeAll(collect); - return result; - } - } - - @PostMapping("/post/rename") - @ResponseStatus(value = HttpStatus.OK) - public List rename(@RequestBody @NonNull List postToRename) { - synchronized (LOCK) { - renamePosts(postToRename); - return postToRename; - } - } - - private void renamePosts(@RequestBody @NonNull List postToRename) { - synchronized (LOCK) { - for (AltPostName altPostName : postToRename) { - try { - pathService.rename( - altPostName.getPostId(), altPostName.getAltName(), settingsService.getSettings()); - } catch (Exception e) { - log.error( - String.format("Failed to rename post with postId = %s", altPostName.getPostId()), e); - throw new ServerErrorException(e.getMessage()); - } - } - } - } - - @PostMapping("/post/rename/first") - @ResponseStatus(value = HttpStatus.OK) - public List renameFirst(@RequestBody @NonNull List postToRename) { - synchronized (LOCK) { - renamePostsToFirst(postToRename); - return postToRename; - } - } - - public void renamePostsToFirst(@RequestBody @NonNull List postToRename) { - synchronized (LOCK) { - for (PostId postId : postToRename) { - Optional _metadata = dataService.findMetadataByPostId(postId.getPostId()); - if (_metadata.isPresent()) { - Metadata metadata = _metadata.get(); - if (metadata.getResolvedNames() != null) { - List resolvedNames = metadata.getResolvedNames(); - if (!resolvedNames.isEmpty()) { - String altTitle = resolvedNames.get(0); - try { - pathService.rename(postId.getPostId(), altTitle, settingsService.getSettings()); - } catch (Exception e) { - log.error( - String.format("Failed to rename post with postId = %s", postId.getPostId()), e); - throw new ServerErrorException(e.getMessage()); - } - } - } - } - } - } - } - - @PostMapping("/post/clear/all") - @ResponseStatus(value = HttpStatus.OK) - public RemoveAllResult clearAll() { - synchronized (LOCK) { - return new RemoveAllResult(dataService.clearCompleted()); - } - } - - @GetMapping("/grab/{threadId}") - @ResponseStatus(value = HttpStatus.OK) - public List grab(@PathVariable("threadId") @NonNull String threadId) { - synchronized (LOCK) { - try { - Queued queued = - dataService - .findQueuedByThreadId(threadId) - .orElseThrow( - () -> - new NotFoundException( - String.format("Unable to find links for threadId = %s", threadId))); - MultiPostScanResult multiPostScanResult = postService.get(queued); - if (multiPostScanResult == null) { - log.error(String.format("Failed to get links for threadId = %s", threadId)); - throw new ServerErrorException( - String.format("Failed to get links for threadId = %s", threadId)); - } else { - return multiPostScanResult.getPosts(); - } - } catch (Exception e) { - throw new ServerErrorException( - String.format("Failed to get links for threadId = %s, %s", threadId, e.getMessage())); - } - } - } - - @PostMapping("/grab/remove") - @ResponseStatus(value = HttpStatus.OK) - public ThreadId grabRemove(@RequestBody @NonNull ThreadId threadId) { - synchronized (LOCK) { - postService.remove(threadId.getThreadId()); - return threadId; - } - } - - @GetMapping("/grab/clear") - @ResponseStatus(value = HttpStatus.OK) - public void grabClear() { - synchronized (LOCK) { - dataService.clearQueueLinks(); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/SettingsRestEndpoint.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/SettingsRestEndpoint.java deleted file mode 100644 index a99887c0..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/SettingsRestEndpoint.java +++ /dev/null @@ -1,66 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints; - -import lombok.extern.slf4j.Slf4j; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.http.HttpStatus; -import org.springframework.web.bind.annotation.*; -import tn.mnlr.vripper.exception.ValidationException; -import tn.mnlr.vripper.services.SettingsService; -import tn.mnlr.vripper.services.domain.Settings; -import tn.mnlr.vripper.web.restendpoints.exceptions.BadRequestException; - -import java.util.List; - -@RestController -@Slf4j -@CrossOrigin(value = "*") -public class SettingsRestEndpoint { - - private final SettingsService settingsService; - - @Autowired - public SettingsRestEndpoint(SettingsService settingsService) { - this.settingsService = settingsService; - } - - @PostMapping("/settings/theme") - @ResponseStatus(value = HttpStatus.OK) - public SettingsService.Theme postTheme(@RequestBody SettingsService.Theme theme) { - this.settingsService.setTheme(theme); - return settingsService.getTheme(); - } - - @GetMapping("/settings/theme") - @ResponseStatus(value = HttpStatus.OK) - public SettingsService.Theme getTheme() { - return settingsService.getTheme(); - } - - @PostMapping("/settings") - @ResponseStatus(value = HttpStatus.OK) - public Settings postSettings(@RequestBody Settings settings) { - - try { - this.settingsService.check(settings); - } catch (ValidationException e) { - log.error("Invalid settings", e); - throw new BadRequestException(e.getMessage()); - } - - this.settingsService.newSettings(settings); - return getAppSettingsService(); - } - - @GetMapping("/settings") - @ResponseStatus(value = HttpStatus.OK) - public Settings getAppSettingsService() { - - return settingsService.getSettings(); - } - - @GetMapping("/settings/proxies") - @ResponseStatus(value = HttpStatus.OK) - public List mirrors() { - return settingsService.getProxies(); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/WebSecurityConfig.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/WebSecurityConfig.java deleted file mode 100644 index f1ce2318..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/WebSecurityConfig.java +++ /dev/null @@ -1,14 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints; - -import org.springframework.context.annotation.Configuration; -import org.springframework.web.servlet.config.annotation.ViewControllerRegistry; -import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; - -@Configuration -public class WebSecurityConfig implements WebMvcConfigurer { - - @Override - public void addViewControllers(ViewControllerRegistry registry) { - registry.addViewController("/home").setViewName("forward:/"); - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/AltPostName.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/AltPostName.java deleted file mode 100644 index 241eaf2d..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/AltPostName.java +++ /dev/null @@ -1,15 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints.domain; - -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; - -@Getter -@Setter -@NoArgsConstructor -@ToString -public class AltPostName { - private String postId; - private String altName; -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/DownloadPath.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/DownloadPath.java deleted file mode 100644 index 5f6eb055..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/DownloadPath.java +++ /dev/null @@ -1,18 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints.domain; - -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; - -@Getter -@Setter -@NoArgsConstructor -@ToString -public class DownloadPath { - private String path; - - public DownloadPath(String path) { - this.path = path; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/PostId.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/PostId.java deleted file mode 100644 index 5079a38b..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/PostId.java +++ /dev/null @@ -1,18 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints.domain; - -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; - -@Getter -@Setter -@NoArgsConstructor -@ToString -public class PostId { - private String postId; - - public PostId(String postId) { - this.postId = postId; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/PostToAdd.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/PostToAdd.java deleted file mode 100644 index aedd9c82..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/PostToAdd.java +++ /dev/null @@ -1,15 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints.domain; - -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; - -@Getter -@Setter -@NoArgsConstructor -@ToString -public class PostToAdd { - private String threadId; - private String postId; -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/RemoveAllResult.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/RemoveAllResult.java deleted file mode 100644 index 1dd91456..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/RemoveAllResult.java +++ /dev/null @@ -1,22 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints.domain; - -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; - -import java.util.List; - -@Getter -@Setter -@NoArgsConstructor -@ToString -public class RemoveAllResult { - private List postIds; - private int removed; - - public RemoveAllResult(List postIds) { - this.removed = postIds.size(); - this.postIds = postIds; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/RemoveResult.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/RemoveResult.java deleted file mode 100644 index 66351dda..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/RemoveResult.java +++ /dev/null @@ -1,18 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints.domain; - -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; - -@Getter -@Setter -@NoArgsConstructor -@ToString -public class RemoveResult { - private String postId; - - public RemoveResult(String postId) { - this.postId = postId; - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/ThreadId.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/ThreadId.java deleted file mode 100644 index a02d08df..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/ThreadId.java +++ /dev/null @@ -1,14 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints.domain; - -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; - -@Getter -@Setter -@NoArgsConstructor -@ToString -public class ThreadId { - private String threadId; -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/ThreadUrl.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/ThreadUrl.java deleted file mode 100644 index 275a1ca0..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/restendpoints/domain/ThreadUrl.java +++ /dev/null @@ -1,14 +0,0 @@ -package tn.mnlr.vripper.web.restendpoints.domain; - -import lombok.Getter; -import lombok.NoArgsConstructor; -import lombok.Setter; -import lombok.ToString; - -@Getter -@Setter -@NoArgsConstructor -@ToString -public class ThreadUrl { - private String url; -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/wsendpoints/DataBroadcast.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/wsendpoints/DataBroadcast.java deleted file mode 100644 index fd157eef..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/wsendpoints/DataBroadcast.java +++ /dev/null @@ -1,158 +0,0 @@ -package tn.mnlr.vripper.web.wsendpoints; - -import lombok.extern.slf4j.Slf4j; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.messaging.simp.SimpMessagingTemplate; -import org.springframework.stereotype.Service; -import reactor.core.Disposable; -import tn.mnlr.vripper.event.Event; -import tn.mnlr.vripper.event.EventBus; -import tn.mnlr.vripper.jpa.domain.Image; -import tn.mnlr.vripper.services.DataService; -import tn.mnlr.vripper.services.domain.DownloadSpeed; -import tn.mnlr.vripper.services.domain.GlobalState; - -import javax.annotation.PostConstruct; -import javax.annotation.PreDestroy; -import java.time.Duration; -import java.time.temporal.ChronoUnit; -import java.util.List; -import java.util.Map; -import java.util.Optional; -import java.util.stream.Collectors; - -@Service -@Slf4j -public class DataBroadcast { - - private final SimpMessagingTemplate template; - private final DataService dataService; - private final EventBus eventBus; - - private Disposable disposable; - - @Autowired - public DataBroadcast(SimpMessagingTemplate template, DataService dataService, EventBus eventBus) { - this.template = template; - this.dataService = dataService; - this.eventBus = eventBus; - } - - @PostConstruct - private void run() { - - disposable = - eventBus - .flux() - .buffer(Duration.of(500, ChronoUnit.MILLIS)) - .subscribe( - data -> { - Map>> eventMap = - data.stream().collect(Collectors.groupingBy(Event::getKind)); - - eventMap.forEach( - (kind, eventList) -> { - switch (kind) { - case POST_UPDATE: - case METADATA_UPDATE: - template.convertAndSend( - "/topic/posts", - eventList.stream() - .map(e -> ((Long) e.getData())) - .distinct() - .map(dataService::findById) - .filter(Optional::isPresent) - .map(Optional::get) - .collect(Collectors.toUnmodifiableSet())); - - break; - case POST_REMOVE: - template.convertAndSend( - "/topic/posts/deleted", - eventList.stream() - .map(e -> ((String) e.getData())) - .collect(Collectors.toUnmodifiableSet())); - break; - case IMAGE_UPDATE: - eventList.stream() - .map(e -> ((Long) e.getData())) - .distinct() - .map(dataService::findImageById) - .filter(Optional::isPresent) - .map(Optional::get) - .collect(Collectors.groupingBy(Image::getPostId)) - .forEach( - (postId, images) -> - template.convertAndSend("/topic/images/" + postId, images)); - break; - - case QUEUED_UPDATE: - template.convertAndSend( - "/topic/queued", - eventList.stream() - .map(e -> ((Long) e.getData())) - .distinct() - .map(dataService::findQueuedById) - .filter(Optional::isPresent) - .map(Optional::get) - .collect(Collectors.toUnmodifiableSet())); - break; - case QUEUED_REMOVE: - template.convertAndSend( - "/topic/queued/deleted", - eventList.stream() - .map(e -> ((String) e.getData())) - .collect(Collectors.toUnmodifiableSet())); - break; - case LOG_EVENT_UPDATE: - template.convertAndSend( - "/topic/events", - eventList.stream() - .map(e -> ((Long) e.getData())) - .distinct() - .map(dataService::findEventById) - .filter(Optional::isPresent) - .map(Optional::get) - .collect(Collectors.toUnmodifiableSet())); - break; - case LOG_EVENT_REMOVE: - template.convertAndSend( - "/topic/events/deleted", - eventList.stream() - .map(e -> ((Long) e.getData())) - .collect(Collectors.toUnmodifiableSet())); - break; - case VG_USER: - eventList.stream() - .map(e -> new DataController.LoggedUser((String) e.getData())) - .distinct() - .forEach(user -> template.convertAndSend("/topic/user", user)); - break; - case GLOBAL_STATE: - eventList.stream() - .map(e -> ((GlobalState) e.getData())) - .distinct() - .forEach( - globalState -> - template.convertAndSend( - "/topic/download-state", globalState)); - break; - case BYTES_PER_SECOND: - eventList.stream() - .map(e -> new DownloadSpeed((Long) e.getData())) - .distinct() - .forEach(speed -> template.convertAndSend("/topic/speed", speed)); - - break; - } - }); - }); - } - - @PreDestroy - private void destroy() { - if (disposable != null) { - disposable.dispose(); - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/wsendpoints/DataController.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/wsendpoints/DataController.java deleted file mode 100644 index 922f2303..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/wsendpoints/DataController.java +++ /dev/null @@ -1,89 +0,0 @@ -package tn.mnlr.vripper.web.wsendpoints; - -import lombok.Getter; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.messaging.handler.annotation.DestinationVariable; -import org.springframework.messaging.simp.annotation.SubscribeMapping; -import org.springframework.stereotype.Controller; -import tn.mnlr.vripper.jpa.domain.Image; -import tn.mnlr.vripper.jpa.domain.LogEvent; -import tn.mnlr.vripper.jpa.domain.Post; -import tn.mnlr.vripper.jpa.domain.Queued; -import tn.mnlr.vripper.services.DataService; -import tn.mnlr.vripper.services.DownloadSpeedService; -import tn.mnlr.vripper.services.GlobalStateService; -import tn.mnlr.vripper.services.VGAuthService; -import tn.mnlr.vripper.services.domain.DownloadSpeed; -import tn.mnlr.vripper.services.domain.GlobalState; - -import java.util.Collection; -import java.util.Comparator; -import java.util.List; - -@Controller -public class DataController { - - private final VGAuthService VGAuthService; - private final GlobalStateService globalStateService; - private final DownloadSpeedService downloadSpeedService; - private final DataService dataService; - - @Autowired - public DataController( - VGAuthService VGAuthService, - GlobalStateService globalStateService, - DownloadSpeedService downloadSpeedService, - DataService dataService) { - this.VGAuthService = VGAuthService; - this.globalStateService = globalStateService; - this.downloadSpeedService = downloadSpeedService; - this.dataService = dataService; - } - - @SubscribeMapping("/user") - public LoggedUser user() { - return new LoggedUser(VGAuthService.getLoggedUser()); - } - - @SubscribeMapping("/download-state") - public GlobalState downloadState() { - return globalStateService.getCurrentState(); - } - - @SubscribeMapping("/speed") - public DownloadSpeed speed() { - return new DownloadSpeed(downloadSpeedService.getCurrentValue()); - } - - @SubscribeMapping("/posts") - public Collection posts() { - List posts = dataService.findAllPosts(); - posts.sort(Comparator.comparing(Post::getAddedOn)); - return posts; - } - - @SubscribeMapping("/images/{postId}") - public List postsDetails(@DestinationVariable("postId") String postId) { - return dataService.findImagesByPostId(postId); - } - - @SubscribeMapping("/queued") - public Collection queued() { - return dataService.findAllQueued(); - } - - @SubscribeMapping("/events") - public Collection events() { - return dataService.findAllEvents(); - } - - @Getter - public static class LoggedUser { - - private final String user; - - LoggedUser(String user) { - this.user = user; - } - } -} diff --git a/vripper-server/src/main/java/tn/mnlr/vripper/web/wsendpoints/WebSocketConfig.java b/vripper-server/src/main/java/tn/mnlr/vripper/web/wsendpoints/WebSocketConfig.java deleted file mode 100644 index 4b682c25..00000000 --- a/vripper-server/src/main/java/tn/mnlr/vripper/web/wsendpoints/WebSocketConfig.java +++ /dev/null @@ -1,23 +0,0 @@ -package tn.mnlr.vripper.web.wsendpoints; - -import org.springframework.context.annotation.Configuration; -import org.springframework.messaging.simp.config.MessageBrokerRegistry; -import org.springframework.web.socket.config.annotation.EnableWebSocketMessageBroker; -import org.springframework.web.socket.config.annotation.StompEndpointRegistry; -import org.springframework.web.socket.config.annotation.WebSocketMessageBrokerConfigurer; - -@Configuration -@EnableWebSocketMessageBroker -public class WebSocketConfig implements WebSocketMessageBrokerConfigurer { - - @Override - public void registerStompEndpoints(StompEndpointRegistry registry) { - registry.addEndpoint("/ws").setAllowedOrigins("*"); - } - - @Override - public void configureMessageBroker(MessageBrokerRegistry registry) { - registry.setApplicationDestinationPrefixes("/app", "/topic"); - registry.enableSimpleBroker("/topic"); - } -} diff --git a/vripper-server/src/main/resources/db/changelog/db.changelog-master.xml b/vripper-server/src/main/resources/db/changelog/db.changelog-master.xml deleted file mode 100644 index fcb349dc..00000000 --- a/vripper-server/src/main/resources/db/changelog/db.changelog-master.xml +++ /dev/null @@ -1,178 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/vripper-server/src/main/resources/logback-spring.xml b/vripper-server/src/main/resources/logback-spring.xml deleted file mode 100644 index 397a6185..00000000 --- a/vripper-server/src/main/resources/logback-spring.xml +++ /dev/null @@ -1,23 +0,0 @@ - - - - - - - - - - - - - - - - - - - - diff --git a/vripper-ui/e2e/protractor.conf.js b/vripper-ui/e2e/protractor.conf.js deleted file mode 100644 index 86776a39..00000000 --- a/vripper-ui/e2e/protractor.conf.js +++ /dev/null @@ -1,28 +0,0 @@ -// Protractor configuration file, see link for more information -// https://github.com/angular/protractor/blob/master/lib/config.ts - -const { SpecReporter } = require('jasmine-spec-reporter'); - -exports.config = { - allScriptsTimeout: 11000, - specs: [ - './src/**/*.e2e-spec.ts' - ], - capabilities: { - 'browserName': 'chrome' - }, - directConnect: true, - baseUrl: 'http://localhost:4200/', - framework: 'jasmine', - jasmineNodeOpts: { - showColors: true, - defaultTimeoutInterval: 30000, - print: function() {} - }, - onPrepare() { - require('ts-node').register({ - project: require('path').join(__dirname, './tsconfig.e2e.json') - }); - jasmine.getEnv().addReporter(new SpecReporter({ spec: { displayStacktrace: true } })); - } -}; \ No newline at end of file diff --git a/vripper-ui/e2e/src/app.e2e-spec.ts b/vripper-ui/e2e/src/app.e2e-spec.ts deleted file mode 100644 index 4f9ab422..00000000 --- a/vripper-ui/e2e/src/app.e2e-spec.ts +++ /dev/null @@ -1,23 +0,0 @@ -import { AppPage } from './app.po'; -import { browser, logging } from 'protractor'; - -describe('workspace-project App', () => { - let page: AppPage; - - beforeEach(() => { - page = new AppPage(); - }); - - it('should display welcome message', () => { - page.navigateTo(); - expect(page.getTitleText()).toEqual('Welcome to vripper-ui!'); - }); - - afterEach(async () => { - // Assert that there are no errors emitted from the browser - const logs = await browser.manage().logs().get(logging.Type.BROWSER); - expect(logs).not.toContain(jasmine.objectContaining({ - level: logging.Level.SEVERE, - })); - }); -}); diff --git a/vripper-ui/e2e/src/app.po.ts b/vripper-ui/e2e/src/app.po.ts deleted file mode 100644 index 72e463a3..00000000 --- a/vripper-ui/e2e/src/app.po.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { browser, by, element } from 'protractor'; - -export class AppPage { - navigateTo() { - return browser.get('/'); - } - - getTitleText() { - return element(by.css('app-root h1')).getText(); - } -} diff --git a/vripper-ui/e2e/tsconfig.e2e.json b/vripper-ui/e2e/tsconfig.e2e.json deleted file mode 100644 index a6dd6220..00000000 --- a/vripper-ui/e2e/tsconfig.e2e.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "extends": "../tsconfig.json", - "compilerOptions": { - "outDir": "../out-tsc/app", - "module": "commonjs", - "target": "es5", - "types": [ - "jasmine", - "jasminewd2", - "node" - ] - } -} \ No newline at end of file diff --git a/vripper-ui/package.json b/vripper-ui/package.json deleted file mode 100644 index 86258173..00000000 --- a/vripper-ui/package.json +++ /dev/null @@ -1,59 +0,0 @@ -{ - "name": "vripper-ui", - "version": "3.5.4", - "scripts": { - "ng": "ng", - "start": "ng serve", - "build": "ng build", - "build-electron": "ng build --prod --base-href ./", - "build-web": "ng build --prod", - "test": "ng test", - "lint": "ng lint", - "e2e": "ng e2e" - }, - "private": true, - "dependencies": { - "@angular/animations": "11.0.8", - "@angular/cdk": "11.0.3", - "@angular/common": "11.0.8", - "@angular/compiler": "11.0.8", - "@angular/core": "11.0.8", - "@angular/flex-layout": "11.0.0-beta.33", - "@angular/forms": "11.0.8", - "@angular/material": "11.0.3", - "@angular/platform-browser": "11.0.8", - "@angular/platform-browser-dynamic": "11.0.8", - "@angular/router": "11.0.8", - "@mdi/angular-material": "^5.8.55", - "@stomp/rx-stomp": "^0.3.5", - "ag-grid-angular": "24.1.0", - "ag-grid-community": "23.2.1", - "core-js": "3.6.5", - "ngx-electron": "2.2.0", - "rxjs": "6.6.3", - "tslib": "^2.0.1", - "zone.js": "0.10.3" - }, - "devDependencies": { - "@angular-devkit/build-angular": "^0.1100.6", - "@angular/cli": "11.0.6", - "@angular/compiler-cli": "11.0.8", - "@angular/language-service": "11.0.8", - "@types/jasmine": "3.5.12", - "@types/jasminewd2": "2.0.8", - "@types/node": "12.12.21", - "codelyzer": "6.0.0", - "electron": "11.1.1", - "jasmine-core": "~3.6.0", - "jasmine-spec-reporter": "~5.0.0", - "karma": "~5.1.1", - "karma-chrome-launcher": "~3.1.0", - "karma-coverage-istanbul-reporter": "~3.0.2", - "karma-jasmine": "~4.0.1", - "karma-jasmine-html-reporter": "^1.5.0", - "protractor": "~7.0.0", - "ts-node": "8.10.2", - "tslint": "~6.1.3", - "typescript": "4.0.5" - } -} diff --git a/vripper-ui/src/app/about/about-component.component-scss-theme.scss b/vripper-ui/src/app/about/about-component.component-scss-theme.scss deleted file mode 100644 index 34a779ab..00000000 --- a/vripper-ui/src/app/about/about-component.component-scss-theme.scss +++ /dev/null @@ -1,10 +0,0 @@ -@import '~@angular/material/theming'; - -@mixin about-component-theme($theme) { - $foreground: map-get($theme, foreground); - $background: map-get($theme, background); - - .about-link, .about-icon { - color: mat-color($foreground, text); - } -} diff --git a/vripper-ui/src/app/app.component.scss-theme.scss b/vripper-ui/src/app/app.component.scss-theme.scss deleted file mode 100644 index b3f21d6b..00000000 --- a/vripper-ui/src/app/app.component.scss-theme.scss +++ /dev/null @@ -1,11 +0,0 @@ -@import '~@angular/material/theming'; - -@mixin app-component-theme($theme) { - - $background: map-get($theme, background); - - button.add-button { - - background-color: mat-color($background, background); - } -} diff --git a/vripper-ui/src/app/domain/download-speed.model.ts b/vripper-ui/src/app/domain/download-speed.model.ts deleted file mode 100644 index 959f3fc6..00000000 --- a/vripper-ui/src/app/domain/download-speed.model.ts +++ /dev/null @@ -1,4 +0,0 @@ -export class DownloadSpeed { - constructor(public speed: string) { - } -} diff --git a/vripper-ui/src/app/domain/logged-user.model.ts b/vripper-ui/src/app/domain/logged-user.model.ts deleted file mode 100644 index ddf66c1f..00000000 --- a/vripper-ui/src/app/domain/logged-user.model.ts +++ /dev/null @@ -1,3 +0,0 @@ -export interface LoggedUser { - user: string -} diff --git a/vripper-ui/src/app/domain/settings.model.ts b/vripper-ui/src/app/domain/settings.model.ts deleted file mode 100644 index 28dfc9c8..00000000 --- a/vripper-ui/src/app/domain/settings.model.ts +++ /dev/null @@ -1,17 +0,0 @@ -export interface Settings { - downloadPath: string; - maxThreads: number; - maxTotalThreads: number; - autoStart: boolean; - vLogin: boolean; - vUsername: string; - vPassword: string; - vThanks: boolean; - desktopClipboard: boolean; - viewPhotos: boolean; - resolveTitle: boolean; - connectionTimeout: number; - maxAttempts: number; - vProxy: string; - maxEventLog: number; -} diff --git a/vripper-ui/src/app/event-log/event-log.component.html b/vripper-ui/src/app/event-log/event-log.component.html deleted file mode 100644 index fdae3ef5..00000000 --- a/vripper-ui/src/app/event-log/event-log.component.html +++ /dev/null @@ -1,4 +0,0 @@ -
- - -
diff --git a/vripper-ui/src/app/multi-post-grid/multi-post-grid.component.html b/vripper-ui/src/app/multi-post-grid/multi-post-grid.component.html deleted file mode 100644 index fdae3ef5..00000000 --- a/vripper-ui/src/app/multi-post-grid/multi-post-grid.component.html +++ /dev/null @@ -1,4 +0,0 @@ -
- - -
diff --git a/vripper-ui/src/app/posts/alternative-title/alternative-title.component.html b/vripper-ui/src/app/posts/alternative-title/alternative-title.component.html deleted file mode 100644 index 77e8c062..00000000 --- a/vripper-ui/src/app/posts/alternative-title/alternative-title.component.html +++ /dev/null @@ -1,26 +0,0 @@ -
-

Alternative title

- - -
- - Choose a title - - - {{altName}} - - - - - - -
-
- - - - -
- diff --git a/vripper-ui/src/app/posts/alternative-title/alternative-title.component.scss b/vripper-ui/src/app/posts/alternative-title/alternative-title.component.scss deleted file mode 100644 index bbfcb12c..00000000 --- a/vripper-ui/src/app/posts/alternative-title/alternative-title.component.scss +++ /dev/null @@ -1,7 +0,0 @@ -form { - height: 100%; -} - -form > mat-form-field { - width: 100%; -} diff --git a/vripper-ui/src/app/posts/alternative-title/alternative-title.component.ts b/vripper-ui/src/app/posts/alternative-title/alternative-title.component.ts deleted file mode 100644 index 8b1a0f2a..00000000 --- a/vripper-ui/src/app/posts/alternative-title/alternative-title.component.ts +++ /dev/null @@ -1,57 +0,0 @@ -import {ChangeDetectionStrategy, Component, Inject, OnInit} from '@angular/core'; -import {MAT_DIALOG_DATA, MatDialogRef} from '@angular/material/dialog'; -import {RenamePostModel} from '../../domain/rename-post.model'; -import {HttpClient} from '@angular/common/http'; -import {ServerService} from '../../services/server-service'; -import {MatSnackBar} from '@angular/material/snack-bar'; -import {FormControl, FormGroup, Validators} from '@angular/forms'; -import {Post} from '../../domain/post-state.model'; - -export interface AlternativeTitleDialog { - post: Post -} - -@Component({ - selector: 'app-alternative-title', - templateUrl: './alternative-title.component.html', - styleUrls: ['./alternative-title.component.scss'], - changeDetection: ChangeDetectionStrategy.OnPush -}) -export class AlternativeTitleComponent implements OnInit { - - form = new FormGroup({ - altNameSelect: new FormControl(''), - altNameInput: new FormControl('', Validators.required) - }); - - constructor( - public dialogRef: MatDialogRef, - @Inject(MAT_DIALOG_DATA) public data: AlternativeTitleDialog, - private httpClient: HttpClient, - private serverService: ServerService, - private _snackBar: MatSnackBar) { - } - - ngOnInit(): void { - this.form.get('altNameSelect').valueChanges.subscribe(v => this.form.get('altNameInput').setValue(v)); - if (this.data.post?.metadata?.resolvedNames == null || this.data.post?.metadata?.resolvedNames.length == 0) { - this.form.get('altNameInput').setValue(this.data.post.title); - } - } - - rename() { - this.httpClient.post(this.serverService.baseUrl + '/post/rename', [{ - postId: this.data.post.postId, - altName: this.form.get('altNameInput').value - }]).subscribe( - () => { - }, - error => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { - duration: 5000 - }); - } - ); - } - -} diff --git a/vripper-ui/src/app/posts/posts.component.html b/vripper-ui/src/app/posts/posts.component.html deleted file mode 100644 index fdae3ef5..00000000 --- a/vripper-ui/src/app/posts/posts.component.html +++ /dev/null @@ -1,4 +0,0 @@ -
- - -
diff --git a/vripper-ui/src/app/services/clipboard.service.ts b/vripper-ui/src/app/services/clipboard.service.ts deleted file mode 100644 index 518a938c..00000000 --- a/vripper-ui/src/app/services/clipboard.service.ts +++ /dev/null @@ -1,72 +0,0 @@ -import {Observable, Subject} from 'rxjs'; -import {ElectronService} from 'ngx-electron'; -import {Injectable} from '@angular/core'; -import {HttpClient} from '@angular/common/http'; -import {Settings} from '../domain/settings.model'; -import {ServerService} from './server-service'; -import Clipboard = Electron.Clipboard; - -@Injectable({ - providedIn: 'root' -}) -export class ClipboardService { - private links$: Subject = new Subject(); - private interval; - private lastText = ''; - - constructor( - private electronService: ElectronService, - private serverService: ServerService, - private httpClient: HttpClient - ) { - } - - get links(): Observable { - return this.links$.asObservable(); - } - - init(settings?: Settings) { - if (!settings) { - this.httpClient.get(this.serverService.baseUrl + '/settings').subscribe( - data => { - this._init(data); - }, - error => { - console.error(error); - } - ); - return; - } else { - this._init(settings); - } - } - - _init(settings: Settings) { - if (!this.electronService.isElectronApp) { - console.log('Clipboard deactivated, not an electron app'); - return; - } - if (this.interval != null) { - clearInterval(this.interval); - } - if (!settings.desktopClipboard) { - return; - } - - const clipboard: Clipboard = this.electronService.clipboard; - this.interval = setInterval(() => { - const text = clipboard.readText(); - - if (this.textHasDiff(text, this.lastText)) { - this.lastText = text; - if (text.indexOf('https://vipergirls.to/threads') !== -1) { - this.links$.next(text); - } - } - }, 500); - } - - private textHasDiff(a, b) { - return a && b !== a; - } -} diff --git a/vripper-ui/src/app/status-bar/status-bar.component.scss-theme.scss b/vripper-ui/src/app/status-bar/status-bar.component.scss-theme.scss deleted file mode 100644 index 05ebd4c0..00000000 --- a/vripper-ui/src/app/status-bar/status-bar.component.scss-theme.scss +++ /dev/null @@ -1,11 +0,0 @@ -@import '~@angular/material/theming'; - -@mixin status-bar-component-theme($theme) { - $foreground: map-get($theme, foreground); - $background: map-get($theme, background); - - .status-bar { - background-color: mat-color($background, background); - color: mat-color($foreground, text); - } -} diff --git a/vripper-ui/src/app/toolbar/toolbar.component.scss-theme.scss b/vripper-ui/src/app/toolbar/toolbar.component.scss-theme.scss deleted file mode 100644 index ddbfe328..00000000 --- a/vripper-ui/src/app/toolbar/toolbar.component.scss-theme.scss +++ /dev/null @@ -1,65 +0,0 @@ -@import '~@angular/material/theming'; - -@mixin toolbar-theme($theme, $dark: false) { - $primary: map-get($theme, primary); - $accent: map-get($theme, accent); - $warn: map-get($theme, warn); - $foreground: map-get($theme, foreground); - $background: map-get($theme, background); - - app-toolbar .mat-toolbar { - @if $dark { - background: mat-color($background); - } @else { - background: mat-color($accent); - } - } - - app-toolbar .mat-form-field-label { - @if $dark { - color: mat-color($foreground, text); - } @else { - color: mat-color($background, background); - } - } - - app-toolbar .mat-icon { - @if $dark { - color: mat-color($foreground, text); - } @else { - color: mat-color($background, background); - } - } - - app-toolbar .mat-form-field { - @if $dark { - color: mat-color($foreground, text); - } @else { - color: mat-color($background, background); - } - } - - app-toolbar .logged-user { - @if $dark { - color: mat-color($foreground, text); - } @else { - color: mat-color($background, background); - } - } - - app-toolbar .mat-icon-button.mat-button-disabled .mat-icon { - @if $dark { - color: mat-color($foreground, disabled-text); - } @else { - color: mat-color($foreground, disabled-text); - } - } - - app-toolbar .add-button .mat-icon { - @if $dark { - color: mat-color($foreground); - } @else { - color: mat-color($accent); - } - } -} diff --git a/vripper-ui/src/dark-theme.scss b/vripper-ui/src/dark-theme.scss deleted file mode 100644 index 34fb901c..00000000 --- a/vripper-ui/src/dark-theme.scss +++ /dev/null @@ -1,9 +0,0 @@ -@import '~@angular/material/theming'; - -$my-dark-theme-primary: mat-palette($mat-grey, 600); -$my-dark-theme-accent: mat-palette($mat-grey, 600); - -$my-dark-theme: mat-dark-theme( - $my-dark-theme-primary, - $my-dark-theme-accent -); diff --git a/vripper-ui/src/grid-theme.scss b/vripper-ui/src/grid-theme.scss deleted file mode 100644 index c36f4937..00000000 --- a/vripper-ui/src/grid-theme.scss +++ /dev/null @@ -1,166 +0,0 @@ -@import '~@angular/material/theming'; - -@import "~ag-grid-community/src/styles/ag-grid.scss"; -@import "~ag-grid-community/src/styles/ag-theme-alpine/sass/ag-theme-alpine-mixin.scss"; - -@mixin grid-theme($theme, $dark: false) { - $primary: map-get($theme, primary); - $accent: map-get($theme, accent); - $warn: map-get($theme, warn); - $foreground: map-get($theme, foreground); - $background: map-get($theme, background); - $red: mat-palette($mat-red); - $grey: mat-palette($mat-grey); - $green: mat-palette($mat-green); - $orange: mat-palette($mat-orange); - $blue: mat-palette($mat-blue); - - .ag-theme-alpine { - @include ag-theme-alpine(( - odd-row-background-color: null, - row-border-color: transparent, - row-hover-color: null, - font-family: (Open, sans-serif), - font-size: 13px - )); - - .ag-checkbox-input-wrapper.ag-checked::after { - color: mat-color($accent); - } - - .ag-checkbox { - margin-right: 5px; - } - - .ag-header-row, .ag-sort-order { - color: mat-color($foreground, text); - } - - .ag-header, .ag-header .ag-icon { - color: mat-color($foreground, text); - background-color: mat-color($background, background); - } - - .ag-root-wrapper { - background-color: mat-color($background, background); - - & .ag-header-cell { - background-color: mat-color($background, background); - } - } - - .ag-row { - background-color: mat-color($background, background); - color: mat-color($foreground, text); - - & a { - color: mat-color($foreground, text); - } - } - - .ag-row-selected { - background-color: transparent; - } - - .ag-center-cols-viewport { - overflow: hidden; - } - - .ag-cell-focus, .ag-cell { - border: none !important; - } - - .ag-row-selected { - background-color: mat-color($background, selected-button) !important; - } - - .text-cell { - white-space: nowrap; - overflow: hidden; - text-overflow: ellipsis; - } - - /* Progress bar section */ - - .native-progress-bar-container { - width: 100%; - height: 100%; - display: flex; - justify-content: start; - align-items: center; - } - - .native-progress-bar-back { - height: 4px; - width: 100%; - background-color: lightgrey; - } - - .native-progress-bar-front { - transition: width 0.75s; - height: 100%; - @if $dark { - background-color: #4a4a4a; - } @else { - background-color: mat-color($primary); - } - } - - /* End progress bar section */ - - /* Status section */ - - .native-status-container { - display: flex; - align-items: center; - justify-content: start; - } - - .native-status-icon { - height: 24px; - margin-right: 5px; - } - - .native-status-text { - line-height: 24px; - } - - .native-status-icon.error { - color: mat-color($red, 400); - } - - .native-status-icon.partial { - color: mat-color($orange, 400); - } - - .native-status-icon.downloading { - color: mat-color($blue, 400); - } - - .native-status-icon.complete { - color: mat-color($green, 400); - } - - .native-status-icon.pending { - color: mat-color($grey, 600); - } - - .native-status-icon.stopped { - color: mat-color($grey, 600); - } - - /* End status section */ - - /* Title section */ - - .cell-icon { - @if $dark { - color: mat-color($accent); - } @else { - color: mat-color($primary); - } - } - - /* End title section */ - } -} diff --git a/vripper-ui/src/karma.conf.js b/vripper-ui/src/karma.conf.js deleted file mode 100644 index ee9caa15..00000000 --- a/vripper-ui/src/karma.conf.js +++ /dev/null @@ -1,31 +0,0 @@ -// Karma configuration file, see link for more information -// https://karma-runner.github.io/1.0/config/configuration-file.html - -module.exports = function (config) { - config.set({ - basePath: '', - frameworks: ['jasmine', '@angular-devkit/build-angular'], - plugins: [ - require('karma-jasmine'), - require('karma-chrome-launcher'), - require('karma-jasmine-html-reporter'), - require('karma-coverage-istanbul-reporter'), - require('@angular-devkit/build-angular/plugins/karma') - ], - client: { - clearContext: false // leave Jasmine Spec Runner output visible in browser - }, - coverageIstanbulReporter: { - dir: require('path').join(__dirname, '../coverage'), - reports: ['html', 'lcovonly', 'text-summary'], - fixWebpackSourcePaths: true - }, - reporters: ['progress', 'kjhtml'], - port: 9876, - colors: true, - logLevel: config.LOG_INFO, - autoWatch: true, - browsers: ['Chrome'], - singleRun: false - }); -}; \ No newline at end of file diff --git a/vripper-ui/src/light-theme.scss b/vripper-ui/src/light-theme.scss deleted file mode 100644 index fb59480a..00000000 --- a/vripper-ui/src/light-theme.scss +++ /dev/null @@ -1,9 +0,0 @@ -@import '~@angular/material/theming'; - -$my-light-theme-primary: mat-palette($mat-red, 900); -$my-light-theme-accent: mat-palette($mat-red, 900); - -$my-light-theme: mat-light-theme( - $my-light-theme-primary, - $my-light-theme-accent -); diff --git a/vripper-ui/src/mat-theme.scss b/vripper-ui/src/mat-theme.scss deleted file mode 100644 index b46aa5c8..00000000 --- a/vripper-ui/src/mat-theme.scss +++ /dev/null @@ -1,11 +0,0 @@ -@import '~@angular/material/theming'; - -$custom-typography: mat-typography-config( - $font-family: 'Open, sans-serif' -); - -@include angular-material-typography($custom-typography); - -@include mat-core($custom-typography); -@import 'light-theme.scss'; -@import 'dark-theme.scss'; diff --git a/vripper-ui/src/polyfills.ts b/vripper-ui/src/polyfills.ts deleted file mode 100644 index c64a485b..00000000 --- a/vripper-ui/src/polyfills.ts +++ /dev/null @@ -1,85 +0,0 @@ -/** - * This file includes polyfills needed by Angular and is loaded before the app. - * You can add your own extra polyfills to this file. - * - * This file is divided into 2 sections: - * 1. Browser polyfills. These are applied before loading ZoneJS and are sorted by browsers. - * 2. Application imports. Files imported after ZoneJS that should be loaded before your main - * file. - * - * The current setup is for so-called "evergreen" browsers; the last versions of browsers that - * automatically update themselves. This includes Safari >= 10, Chrome >= 55 (including Opera), - * Edge >= 13 on the desktop, and iOS 10 and Chrome on mobile. - * - * Learn more in https://angular.io/guide/browser-support - */ - -/*************************************************************************************************** - * BROWSER POLYFILLS - */ - -/** IE9, IE10, IE11, and Chrome <55 requires all of the following polyfills. - * This also includes Android Emulators with older versions of Chrome and Google Search/Googlebot - */ - -// import 'core-js/es6/symbol'; -// import 'core-js/es6/object'; -// import 'core-js/es6/function'; -// import 'core-js/es6/parse-int'; -// import 'core-js/es6/parse-float'; -// import 'core-js/es6/number'; -// import 'core-js/es6/math'; -// import 'core-js/es6/string'; -// import 'core-js/es6/date'; -// import 'core-js/es6/array'; -// import 'core-js/es6/regexp'; -// import 'core-js/es6/map'; -// import 'core-js/es6/weak-map'; -// import 'core-js/es6/set'; - -/** IE10 and IE11 requires the following for NgClass support on SVG elements */ -// import 'classlist.js'; // Run `npm install --save classlist.js`. - -/** IE10 and IE11 requires the following for the Reflect API. */ -// import 'core-js/es6/reflect'; - -/** - * Web Animations `@angular/platform-browser/animations` - * Only required if AnimationBuilder is used within the application and using IE/Edge or Safari. - * Standard animation support in Angular DOES NOT require any polyfills (as of Angular 6.0). - */ -// import 'web-animations-js'; // Run `npm install --save web-animations-js`. - -/** - * By default, zone.js will patch all possible macroTask and DomEvents - * user can disable parts of macroTask/DomEvents patch by setting following flags - * because those flags need to be set before `zone.js` being loaded, and webpack - * will put import in the top of bundle, so user need to create a separate file - * in this directory (for example: zone-flags.ts), and put the following flags - * into that file, and then add the following code before importing zone.js. - * import './zone-flags.ts'; - * - * The flags allowed in zone-flags.ts are listed here. - * - * The following flags will work for all browsers. - * - * (window as any).__Zone_disable_requestAnimationFrame = true; // disable patch requestAnimationFrame - * (window as any).__Zone_disable_on_property = true; // disable patch onProperty such as onclick - * (window as any).__zone_symbol__BLACK_LISTED_EVENTS = ['scroll', 'mousemove']; // disable patch specified eventNames - * - * in IE/Edge developer tools, the addEventListener will also be wrapped by zone.js - * with the following flag, it will bypass `zone.js` patch for IE/Edge - * - * (window as any).__Zone_enable_cross_context_check = true; - * - */ - -/*************************************************************************************************** - * Zone JS is required by default for Angular itself. - */ -import 'zone.js/dist/zone'; // Included with Angular CLI. - - -/*************************************************************************************************** - * APPLICATION IMPORTS - */ diff --git a/vripper-ui/src/test.ts b/vripper-ui/src/test.ts deleted file mode 100644 index ee6a3631..00000000 --- a/vripper-ui/src/test.ts +++ /dev/null @@ -1,17 +0,0 @@ -// This file is required by karma.conf.js and loads recursively all the .spec and framework files - -import 'zone.js/dist/zone-testing'; -import {getTestBed} from '@angular/core/testing'; -import {BrowserDynamicTestingModule, platformBrowserDynamicTesting} from '@angular/platform-browser-dynamic/testing'; - -declare const require: any; - -// First, initialize the Angular testing environment. -getTestBed().initTestEnvironment( - BrowserDynamicTestingModule, - platformBrowserDynamicTesting() -); -// Then we find all the tests. -const context = require.context('./', true, /\.spec\.ts$/); -// And load the modules. -context.keys().map(context); diff --git a/vripper-ui/src/tsconfig.app.json b/vripper-ui/src/tsconfig.app.json deleted file mode 100644 index 6af6e87a..00000000 --- a/vripper-ui/src/tsconfig.app.json +++ /dev/null @@ -1,11 +0,0 @@ -{ - "extends": "../tsconfig.json", - "compilerOptions": { - "outDir": "../out-tsc/app", - "types": [] - }, - "files": [ - "main.ts", - "polyfills.ts" - ] -} diff --git a/vripper-ui/src/tsconfig.spec.json b/vripper-ui/src/tsconfig.spec.json deleted file mode 100644 index de773363..00000000 --- a/vripper-ui/src/tsconfig.spec.json +++ /dev/null @@ -1,18 +0,0 @@ -{ - "extends": "../tsconfig.json", - "compilerOptions": { - "outDir": "../out-tsc/spec", - "types": [ - "jasmine", - "node" - ] - }, - "files": [ - "test.ts", - "polyfills.ts" - ], - "include": [ - "**/*.spec.ts", - "**/*.d.ts" - ] -} diff --git a/vripper-ui/src/tslint.json b/vripper-ui/src/tslint.json deleted file mode 100644 index aa7c3eeb..00000000 --- a/vripper-ui/src/tslint.json +++ /dev/null @@ -1,17 +0,0 @@ -{ - "extends": "../tslint.json", - "rules": { - "directive-selector": [ - true, - "attribute", - "app", - "camelCase" - ], - "component-selector": [ - true, - "element", - "app", - "kebab-case" - ] - } -} diff --git a/vripper-ui/tsconfig.json b/vripper-ui/tsconfig.json deleted file mode 100644 index e4f4ac9d..00000000 --- a/vripper-ui/tsconfig.json +++ /dev/null @@ -1,22 +0,0 @@ -{ - "compileOnSave": false, - "compilerOptions": { - "baseUrl": "./", - "outDir": "./dist/out-tsc", - "sourceMap": true, - "declaration": false, - "module": "es2020", - "moduleResolution": "node", - "emitDecoratorMetadata": true, - "experimentalDecorators": true, - "importHelpers": true, - "target": "es5", - "typeRoots": [ - "node_modules/@types" - ], - "lib": [ - "es2018", - "dom" - ] - } -} diff --git a/vripper-ui/tslint.json b/vripper-ui/tslint.json deleted file mode 100644 index 43700156..00000000 --- a/vripper-ui/tslint.json +++ /dev/null @@ -1,130 +0,0 @@ -{ - "rulesDirectory": [ - "codelyzer" - ], - "rules": { - "arrow-return-shorthand": true, - "callable-types": true, - "class-name": true, - "comment-format": [ - true, - "check-space" - ], - "curly": true, - "deprecation": { - "severity": "warn" - }, - "eofline": true, - "forin": true, - "import-blacklist": [ - true, - "rxjs/Rx" - ], - "import-spacing": true, - "indent": [ - true, - "spaces" - ], - "interface-over-type-literal": true, - "label-position": true, - "max-line-length": [ - true, - 140 - ], - "member-access": false, - "member-ordering": [ - true, - { - "order": [ - "static-field", - "instance-field", - "static-method", - "instance-method" - ] - } - ], - "no-arg": true, - "no-bitwise": true, - "no-console": [ - true, - "debug", - "info", - "time", - "timeEnd", - "trace" - ], - "no-construct": true, - "no-debugger": true, - "no-duplicate-super": true, - "no-empty": false, - "no-empty-interface": true, - "no-eval": true, - "no-inferrable-types": [ - true, - "ignore-params" - ], - "no-misused-new": true, - "no-non-null-assertion": true, - "no-redundant-jsdoc": true, - "no-shadowed-variable": true, - "no-string-literal": false, - "no-string-throw": true, - "no-switch-case-fall-through": true, - "no-trailing-whitespace": true, - "no-unnecessary-initializer": true, - "no-unused-expression": true, - "no-var-keyword": true, - "object-literal-sort-keys": false, - "one-line": [ - true, - "check-open-brace", - "check-catch", - "check-else", - "check-whitespace" - ], - "prefer-const": true, - "quotemark": [ - true, - "single" - ], - "radix": true, - "semicolon": [ - true, - "always" - ], - "triple-equals": [ - true, - "allow-null-check" - ], - "typedef-whitespace": [ - true, - { - "call-signature": "nospace", - "index-signature": "nospace", - "parameter": "nospace", - "property-declaration": "nospace", - "variable-declaration": "nospace" - } - ], - "unified-signatures": true, - "variable-name": false, - "whitespace": [ - true, - "check-branch", - "check-decl", - "check-operator", - "check-separator", - "check-type" - ], - "no-output-on-prefix": true, - "use-input-property-decorator": true, - "use-output-property-decorator": true, - "use-host-property-decorator": true, - "no-input-rename": true, - "no-output-rename": true, - "use-life-cycle-interface": true, - "use-pipe-transform-interface": true, - "component-class-suffix": true, - "directive-class-suffix": true - } -} diff --git a/vripper-ui/yarn.lock b/vripper-ui/yarn.lock deleted file mode 100644 index ef3d4d04..00000000 --- a/vripper-ui/yarn.lock +++ /dev/null @@ -1,9500 +0,0 @@ -# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. -# yarn lockfile v1 - - -"@angular-devkit/architect@0.1100.6": - version "0.1100.6" - resolved "https://registry.yarnpkg.com/@angular-devkit/architect/-/architect-0.1100.6.tgz#ce90ffb78d1d945cafc339d4cfc63b3582cb8e6a" - integrity sha512-4O+cg3AimI2bNAxxdu5NrqSf4Oa8r8xL0+G2Ycd3jLoFv0h0ecJiNKEG5F6IpTprb4aexZD6pcxBJCqQ8MmzWQ== - dependencies: - "@angular-devkit/core" "11.0.6" - rxjs "6.6.3" - -"@angular-devkit/architect@0.1100.7": - version "0.1100.7" - resolved "https://registry.yarnpkg.com/@angular-devkit/architect/-/architect-0.1100.7.tgz#6c39b886bbdb5d0e5a2d9921aa5816209041e105" - integrity sha512-b2zv2yiRbdhJ7hJfZsAvGYcqgh2DVtc7gRIPo1eDPvOAKrenmZ4zo/v0PRYScrTsPzqmoCokNA5nIwufwUEnuA== - dependencies: - "@angular-devkit/core" "11.0.7" - rxjs "6.6.3" - -"@angular-devkit/build-angular@^0.1100.6": - version "0.1100.7" - resolved "https://registry.yarnpkg.com/@angular-devkit/build-angular/-/build-angular-0.1100.7.tgz#e5577998e67e89137c2b7c9d232a58b2fdca91dc" - integrity sha512-erc+AtSU46ZIX7A5dmeZ0/G/SQIbqMAGbTKZbf11GePyhT0JAAnfMQtOHMb6AaX85n4yQTg1uMo9f5+8V3lfKA== - dependencies: - "@angular-devkit/architect" "0.1100.7" - "@angular-devkit/build-optimizer" "0.1100.7" - "@angular-devkit/build-webpack" "0.1100.7" - "@angular-devkit/core" "11.0.7" - "@babel/core" "7.12.3" - "@babel/generator" "7.12.1" - "@babel/plugin-transform-runtime" "7.12.1" - "@babel/preset-env" "7.12.1" - "@babel/runtime" "7.12.1" - "@babel/template" "7.10.4" - "@jsdevtools/coverage-istanbul-loader" "3.0.5" - "@ngtools/webpack" "11.0.7" - ansi-colors "4.1.1" - autoprefixer "9.8.6" - babel-loader "8.1.0" - browserslist "^4.9.1" - cacache "15.0.5" - caniuse-lite "^1.0.30001032" - circular-dependency-plugin "5.2.0" - copy-webpack-plugin "6.2.1" - core-js "3.6.5" - css-loader "4.3.0" - cssnano "4.1.10" - file-loader "6.1.1" - find-cache-dir "3.3.1" - glob "7.1.6" - inquirer "7.3.3" - jest-worker "26.5.0" - karma-source-map-support "1.4.0" - less "3.12.2" - less-loader "7.0.2" - license-webpack-plugin "2.3.1" - loader-utils "2.0.0" - mini-css-extract-plugin "1.2.1" - minimatch "3.0.4" - open "7.3.0" - ora "5.1.0" - parse5-html-rewriting-stream "6.0.1" - pnp-webpack-plugin "1.6.4" - postcss "7.0.32" - postcss-import "12.0.1" - postcss-loader "4.0.4" - raw-loader "4.0.2" - regenerator-runtime "0.13.7" - resolve-url-loader "3.1.2" - rimraf "3.0.2" - rollup "2.32.1" - rxjs "6.6.3" - sass "1.27.0" - sass-loader "10.0.5" - semver "7.3.2" - source-map "0.7.3" - source-map-loader "1.1.2" - source-map-support "0.5.19" - speed-measure-webpack-plugin "1.3.3" - style-loader "2.0.0" - stylus "0.54.8" - stylus-loader "4.3.1" - terser "5.3.7" - terser-webpack-plugin "4.2.3" - text-table "0.2.0" - tree-kill "1.2.2" - webpack "4.44.2" - webpack-dev-middleware "3.7.2" - webpack-dev-server "3.11.0" - webpack-merge "5.2.0" - webpack-sources "2.0.1" - webpack-subresource-integrity "1.5.1" - worker-plugin "5.0.0" - -"@angular-devkit/build-optimizer@0.1100.7": - version "0.1100.7" - resolved "https://registry.yarnpkg.com/@angular-devkit/build-optimizer/-/build-optimizer-0.1100.7.tgz#fef39c8db5d178f223d43bd1c6e3a566e19c24c1" - integrity sha512-bHIIub0d1trVAmAX/EaNR6Zo4b7hkscewK394qYYp/w8VKQkLSAPMUbt2YTWN+erR9yyHnJ2y7tBabIui75Wdw== - dependencies: - loader-utils "2.0.0" - source-map "0.7.3" - tslib "2.0.3" - typescript "4.0.5" - webpack-sources "2.0.1" - -"@angular-devkit/build-webpack@0.1100.7": - version "0.1100.7" - resolved "https://registry.yarnpkg.com/@angular-devkit/build-webpack/-/build-webpack-0.1100.7.tgz#d62df6bd70b495df7dc335f80ecd925e0f3b3e4d" - integrity sha512-/6Hudd1hs/GMHX4C/Qk7jueIMNg8NKFJWDEbvMPMgDzTqUIa680PTD6SNSCcY5Cz9mEpdpYCZo5N31JB7dlpOg== - dependencies: - "@angular-devkit/architect" "0.1100.7" - "@angular-devkit/core" "11.0.7" - rxjs "6.6.3" - -"@angular-devkit/core@11.0.6": - version "11.0.6" - resolved "https://registry.yarnpkg.com/@angular-devkit/core/-/core-11.0.6.tgz#b3ea815ecdea5f77dae58f3f410b268453d7eb8b" - integrity sha512-nhvU5hH01r9qcexAqvIFU233treWWeW3ncs9UFYjD9Hys9sDSvqC3+bvGvl9vCG5FsyY7oDsjaVAipyUc+SFAg== - dependencies: - ajv "6.12.6" - fast-json-stable-stringify "2.1.0" - magic-string "0.25.7" - rxjs "6.6.3" - source-map "0.7.3" - -"@angular-devkit/core@11.0.7": - version "11.0.7" - resolved "https://registry.yarnpkg.com/@angular-devkit/core/-/core-11.0.7.tgz#05b82be7e11ddc2a7836c777297ee45eee20ff55" - integrity sha512-1GKnIT++YSUHpzzRx9QC0+8yOw4wy+ZpiJVDlroPSeK4FGrTCJqJKenkfRjVFRFOSrzTiJds+IU6kI4+bFbw9g== - dependencies: - ajv "6.12.6" - fast-json-stable-stringify "2.1.0" - magic-string "0.25.7" - rxjs "6.6.3" - source-map "0.7.3" - -"@angular-devkit/schematics@11.0.6": - version "11.0.6" - resolved "https://registry.yarnpkg.com/@angular-devkit/schematics/-/schematics-11.0.6.tgz#06631190cb22609462597cd6659080fc3313582a" - integrity sha512-hCyu/SSSiC6dKl/NxdWctknIrBqKR6pRe7DMArWowrZX6P9oi36LpKEFnKutE8+tXjsOqQj8XMBq9L64sXZWqg== - dependencies: - "@angular-devkit/core" "11.0.6" - ora "5.1.0" - rxjs "6.6.3" - -"@angular/animations@11.0.8": - version "11.0.8" - resolved "https://registry.yarnpkg.com/@angular/animations/-/animations-11.0.8.tgz#abb3a016bc9164176d1edfcbd2783cc3ba6f9ca3" - integrity sha512-LlKR5bO7WsykZGcPDnL7c4q2/eq2fqywD+KMw5FemGLI8mN1mnq30br9/8TgaFwY0Do1OTOUh1JSrn1ybQgTwA== - dependencies: - tslib "^2.0.0" - -"@angular/cdk@11.0.3": - version "11.0.3" - resolved "https://registry.yarnpkg.com/@angular/cdk/-/cdk-11.0.3.tgz#f5a119e28b4ba21a6f013dfd4de41b16083bd25a" - integrity sha512-hgbJXvZURKBnZawwxUrsZE/3a+HCJh2UhoLIng3cn5Q+WIW/4a37knDl8B9DYKBWrCqeINXNcUHVSKkWc/gjCA== - dependencies: - tslib "^2.0.0" - optionalDependencies: - parse5 "^5.0.0" - -"@angular/cli@11.0.6": - version "11.0.6" - resolved "https://registry.yarnpkg.com/@angular/cli/-/cli-11.0.6.tgz#8d65d3ad3841aabe23ff38a41fa6c4f38dd12f66" - integrity sha512-bwrXXyU23HjUlFl0CNCU+XMGa/enooqpMLcTAA15StVpKFHyaA4c57il/aqu+1IuB+zR6rGDzhAABuvRcHd+mQ== - dependencies: - "@angular-devkit/architect" "0.1100.6" - "@angular-devkit/core" "11.0.6" - "@angular-devkit/schematics" "11.0.6" - "@schematics/angular" "11.0.6" - "@schematics/update" "0.1100.6" - "@yarnpkg/lockfile" "1.1.0" - ansi-colors "4.1.1" - debug "4.2.0" - ini "1.3.6" - inquirer "7.3.3" - npm-package-arg "8.1.0" - npm-pick-manifest "6.1.0" - open "7.3.0" - pacote "9.5.12" - resolve "1.18.1" - rimraf "3.0.2" - semver "7.3.2" - symbol-observable "2.0.3" - universal-analytics "0.4.23" - uuid "8.3.1" - -"@angular/common@11.0.8": - version "11.0.8" - resolved "https://registry.yarnpkg.com/@angular/common/-/common-11.0.8.tgz#f3fd177af584a4e59b2a9bc1a90fa45dcb3dd672" - integrity sha512-Or7uSetk29wcKOsDVyJl/2JUC34e/gDNI3HNlpFh98miT8G4tqFKSmuLGRPPanIKqyQQQmquV93VNPOMA+rdYA== - dependencies: - tslib "^2.0.0" - -"@angular/compiler-cli@11.0.8": - version "11.0.8" - resolved "https://registry.yarnpkg.com/@angular/compiler-cli/-/compiler-cli-11.0.8.tgz#4284fc6e9cc7bee5314738cd32ce8bbdba20237c" - integrity sha512-xblOhSgshNzAxNmLdBjdLHMc3Zoc12w5Bz70nrqtAfK9DY6kQx6aAxpbhOr/F5Jl5XFNbvIviOOkG9aqM1Iuuw== - dependencies: - "@babel/core" "^7.8.6" - "@babel/types" "^7.8.6" - canonical-path "1.0.0" - chokidar "^3.0.0" - convert-source-map "^1.5.1" - dependency-graph "^0.7.2" - fs-extra "4.0.2" - magic-string "^0.25.0" - minimist "^1.2.0" - reflect-metadata "^0.1.2" - semver "^6.3.0" - source-map "^0.6.1" - sourcemap-codec "^1.4.8" - tslib "^2.0.0" - yargs "^16.1.1" - -"@angular/compiler@11.0.8": - version "11.0.8" - resolved "https://registry.yarnpkg.com/@angular/compiler/-/compiler-11.0.8.tgz#da59389d805034a261b839d703bca3134b6157c4" - integrity sha512-v7NjxLvyJl2cxH7roVTMLnaFCLA9gStgndxJcsXtD+hI7hCOFvrzxwpm3J822KYEkwWhB1PMthJwrZ6OWtkp/A== - dependencies: - tslib "^2.0.0" - -"@angular/compiler@9.0.0": - version "9.0.0" - resolved "https://registry.yarnpkg.com/@angular/compiler/-/compiler-9.0.0.tgz#87e0bef4c369b6cadae07e3a4295778fc93799d5" - integrity sha512-ctjwuntPfZZT2mNj2NDIVu51t9cvbhl/16epc5xEwyzyDt76pX9UgwvY+MbXrf/C/FWwdtmNtfP698BKI+9leQ== - -"@angular/core@11.0.8": - version "11.0.8" - resolved "https://registry.yarnpkg.com/@angular/core/-/core-11.0.8.tgz#cc4ffc5506b393bf851fd7b94760d2a66efeb47c" - integrity sha512-NRKCgOtPFKcJpjjSV0NKuoNaD6ZQ3RqXBm9JURbnjVLaTwtCJlQ9H3N/e+HZNw8Ha2TDFJoiYX+RY071cyTMHQ== - dependencies: - tslib "^2.0.0" - -"@angular/core@9.0.0": - version "9.0.0" - resolved "https://registry.yarnpkg.com/@angular/core/-/core-9.0.0.tgz#227dc53e1ac81824f998c6e76000b7efc522641e" - integrity sha512-6Pxgsrf0qF9iFFqmIcWmjJGkkCaCm6V5QNnxMy2KloO3SDq6QuMVRbN9RtC8Urmo25LP+eZ6ZgYqFYpdD8Hd9w== - -"@angular/flex-layout@11.0.0-beta.33": - version "11.0.0-beta.33" - resolved "https://registry.yarnpkg.com/@angular/flex-layout/-/flex-layout-11.0.0-beta.33.tgz#c2d08f90164701b66a62b8c4365b4e22c1e95789" - integrity sha512-unfhw3abZuKtdwQicRStHCYGbANPTHYg4WNRQk/RC5Mxq+4WOp4Q8HI7GqRHCGUYDCGUP7w1sU/oDt8f09nM8w== - dependencies: - tslib "^2.0.0" - -"@angular/forms@11.0.8": - version "11.0.8" - resolved "https://registry.yarnpkg.com/@angular/forms/-/forms-11.0.8.tgz#f0d68407b50174e4c8b6d3b134fafed792db5320" - integrity sha512-soTg9Zc3G08J/+LRq5FxFvVsKoDQb4F29nEVpbH7oK2hof9Qqois9lH8bJy5dHbNh1qNXYmlWjUd7CQ9p+bbUg== - dependencies: - tslib "^2.0.0" - -"@angular/language-service@11.0.8": - version "11.0.8" - resolved "https://registry.yarnpkg.com/@angular/language-service/-/language-service-11.0.8.tgz#2354589115c7f851c016c7d3a68952f37040b80c" - integrity sha512-5iKvCzb0YWCrEvCUCGsmwQxfUQjnAOdeaNHieLmfkpAutojtIrK59rNmyQqA2RBpo2OZMANz0/aOde1N2rHI0w== - -"@angular/material@11.0.3": - version "11.0.3" - resolved "https://registry.yarnpkg.com/@angular/material/-/material-11.0.3.tgz#dd848f1804b07c9654b095d5b1accb13083e91e5" - integrity sha512-YTHmtKGwjAEFAOOmuivcwnINMPHxvM7iZQpTgZqDKNiqiv53cwry2Ctb54suRNT+D794z59D0/r+YocGkzFv3w== - dependencies: - tslib "^2.0.0" - -"@angular/platform-browser-dynamic@11.0.8": - version "11.0.8" - resolved "https://registry.yarnpkg.com/@angular/platform-browser-dynamic/-/platform-browser-dynamic-11.0.8.tgz#f7714f40e0cb3044f7dfd362371d39c9c4501b19" - integrity sha512-rirFL6R0R9wXTYfzZzsF7Zchq48439QQHAQ3Pi0nTDtyn3seHIEWMW1FcwjuPIt/ea5F+kRB4L8fxGK8G1/taA== - dependencies: - tslib "^2.0.0" - -"@angular/platform-browser@11.0.8": - version "11.0.8" - resolved "https://registry.yarnpkg.com/@angular/platform-browser/-/platform-browser-11.0.8.tgz#aa3863512ad63373d9c6d13ecb64cc1357de959e" - integrity sha512-07nbExdtf2fRJh3XEicbCj1fK0/tWmIW/TaWJ9hv1xeQLjPDcplTF0MaPSlow36qyO+q83rsd3hxyFfBTqgOyw== - dependencies: - tslib "^2.0.0" - -"@angular/router@11.0.8": - version "11.0.8" - resolved "https://registry.yarnpkg.com/@angular/router/-/router-11.0.8.tgz#771b4a235eb696d4411e867f13701532b0d95c67" - integrity sha512-vinpCct3r+6wFtGdYNa/l5PCpJqQSB/D9Z+zZBdmDUy9s32dggHG4msmVrYbFH/j7cKDAycqHGVFyj0hKGDQiQ== - dependencies: - tslib "^2.0.0" - -"@babel/code-frame@^7.0.0", "@babel/code-frame@^7.10.4", "@babel/code-frame@^7.12.13": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.12.13.tgz#dcfc826beef65e75c50e21d3837d7d95798dd658" - integrity sha512-HV1Cm0Q3ZrpCR93tkWOYiuYIgLxZXZFVG2VgK+MBWjUqZTundupbfx2aXarXuw5Ko5aMcjtJgbSs4vUGBS5v6g== - dependencies: - "@babel/highlight" "^7.12.13" - -"@babel/compat-data@^7.12.1", "@babel/compat-data@^7.13.8": - version "7.13.8" - resolved "https://registry.yarnpkg.com/@babel/compat-data/-/compat-data-7.13.8.tgz#5b783b9808f15cef71547f1b691f34f8ff6003a6" - integrity sha512-EaI33z19T4qN3xLXsGf48M2cDqa6ei9tPZlfLdb2HC+e/cFtREiRd8hdSqDbwdLB0/+gLwqJmCYASH0z2bUdog== - -"@babel/core@7.12.3": - version "7.12.3" - resolved "https://registry.yarnpkg.com/@babel/core/-/core-7.12.3.tgz#1b436884e1e3bff6fb1328dc02b208759de92ad8" - integrity sha512-0qXcZYKZp3/6N2jKYVxZv0aNCsxTSVCiK72DTiTYZAu7sjg73W0/aynWjMbiGd87EQL4WyA8reiJVh92AVla9g== - dependencies: - "@babel/code-frame" "^7.10.4" - "@babel/generator" "^7.12.1" - "@babel/helper-module-transforms" "^7.12.1" - "@babel/helpers" "^7.12.1" - "@babel/parser" "^7.12.3" - "@babel/template" "^7.10.4" - "@babel/traverse" "^7.12.1" - "@babel/types" "^7.12.1" - convert-source-map "^1.7.0" - debug "^4.1.0" - gensync "^1.0.0-beta.1" - json5 "^2.1.2" - lodash "^4.17.19" - resolve "^1.3.2" - semver "^5.4.1" - source-map "^0.5.0" - -"@babel/core@^7.7.5", "@babel/core@^7.8.6": - version "7.13.10" - resolved "https://registry.yarnpkg.com/@babel/core/-/core-7.13.10.tgz#07de050bbd8193fcd8a3c27918c0890613a94559" - integrity sha512-bfIYcT0BdKeAZrovpMqX2Mx5NrgAckGbwT982AkdS5GNfn3KMGiprlBAtmBcFZRUmpaufS6WZFP8trvx8ptFDw== - dependencies: - "@babel/code-frame" "^7.12.13" - "@babel/generator" "^7.13.9" - "@babel/helper-compilation-targets" "^7.13.10" - "@babel/helper-module-transforms" "^7.13.0" - "@babel/helpers" "^7.13.10" - "@babel/parser" "^7.13.10" - "@babel/template" "^7.12.13" - "@babel/traverse" "^7.13.0" - "@babel/types" "^7.13.0" - convert-source-map "^1.7.0" - debug "^4.1.0" - gensync "^1.0.0-beta.2" - json5 "^2.1.2" - lodash "^4.17.19" - semver "^6.3.0" - source-map "^0.5.0" - -"@babel/generator@7.12.1": - version "7.12.1" - resolved "https://registry.yarnpkg.com/@babel/generator/-/generator-7.12.1.tgz#0d70be32bdaa03d7c51c8597dda76e0df1f15468" - integrity sha512-DB+6rafIdc9o72Yc3/Ph5h+6hUjeOp66pF0naQBgUFFuPqzQwIlPTm3xZR7YNvduIMtkDIj2t21LSQwnbCrXvg== - dependencies: - "@babel/types" "^7.12.1" - jsesc "^2.5.1" - source-map "^0.5.0" - -"@babel/generator@^7.12.1", "@babel/generator@^7.13.0", "@babel/generator@^7.13.9": - version "7.13.9" - resolved "https://registry.yarnpkg.com/@babel/generator/-/generator-7.13.9.tgz#3a7aa96f9efb8e2be42d38d80e2ceb4c64d8de39" - integrity sha512-mHOOmY0Axl/JCTkxTU6Lf5sWOg/v8nUa+Xkt4zMTftX0wqmb6Sh7J8gvcehBw7q0AhrhAR+FDacKjCZ2X8K+Sw== - dependencies: - "@babel/types" "^7.13.0" - jsesc "^2.5.1" - source-map "^0.5.0" - -"@babel/helper-annotate-as-pure@^7.12.13": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.12.13.tgz#0f58e86dfc4bb3b1fcd7db806570e177d439b6ab" - integrity sha512-7YXfX5wQ5aYM/BOlbSccHDbuXXFPxeoUmfWtz8le2yTkTZc+BxsiEnENFoi2SlmA8ewDkG2LgIMIVzzn2h8kfw== - dependencies: - "@babel/types" "^7.12.13" - -"@babel/helper-builder-binary-assignment-operator-visitor@^7.12.13": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.12.13.tgz#6bc20361c88b0a74d05137a65cac8d3cbf6f61fc" - integrity sha512-CZOv9tGphhDRlVjVkAgm8Nhklm9RzSmWpX2my+t7Ua/KT616pEzXsQCjinzvkRvHWJ9itO4f296efroX23XCMA== - dependencies: - "@babel/helper-explode-assignable-expression" "^7.12.13" - "@babel/types" "^7.12.13" - -"@babel/helper-compilation-targets@^7.12.1", "@babel/helper-compilation-targets@^7.13.10", "@babel/helper-compilation-targets@^7.13.8": - version "7.13.10" - resolved "https://registry.yarnpkg.com/@babel/helper-compilation-targets/-/helper-compilation-targets-7.13.10.tgz#1310a1678cb8427c07a753750da4f8ce442bdd0c" - integrity sha512-/Xju7Qg1GQO4mHZ/Kcs6Au7gfafgZnwm+a7sy/ow/tV1sHeraRUHbjdat8/UvDor4Tez+siGKDk6zIKtCPKVJA== - dependencies: - "@babel/compat-data" "^7.13.8" - "@babel/helper-validator-option" "^7.12.17" - browserslist "^4.14.5" - semver "^6.3.0" - -"@babel/helper-create-class-features-plugin@^7.13.0": - version "7.13.10" - resolved "https://registry.yarnpkg.com/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.13.10.tgz#073b2bbb925a097643c6fc5770e5f13394e887c9" - integrity sha512-YV7r2YxdTUaw84EwNkyrRke/TJHR/UXGiyvACRqvdVJ2/syV2rQuJNnaRLSuYiop8cMRXOgseTGoJCWX0q2fFg== - dependencies: - "@babel/helper-function-name" "^7.12.13" - "@babel/helper-member-expression-to-functions" "^7.13.0" - "@babel/helper-optimise-call-expression" "^7.12.13" - "@babel/helper-replace-supers" "^7.13.0" - "@babel/helper-split-export-declaration" "^7.12.13" - -"@babel/helper-create-regexp-features-plugin@^7.12.13": - version "7.12.17" - resolved "https://registry.yarnpkg.com/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.12.17.tgz#a2ac87e9e319269ac655b8d4415e94d38d663cb7" - integrity sha512-p2VGmBu9oefLZ2nQpgnEnG0ZlRPvL8gAGvPUMQwUdaE8k49rOMuZpOwdQoy5qJf6K8jL3bcAMhVUlHAjIgJHUg== - dependencies: - "@babel/helper-annotate-as-pure" "^7.12.13" - regexpu-core "^4.7.1" - -"@babel/helper-explode-assignable-expression@^7.12.13": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.13.0.tgz#17b5c59ff473d9f956f40ef570cf3a76ca12657f" - integrity sha512-qS0peLTDP8kOisG1blKbaoBg/o9OSa1qoumMjTK5pM+KDTtpxpsiubnCGP34vK8BXGcb2M9eigwgvoJryrzwWA== - dependencies: - "@babel/types" "^7.13.0" - -"@babel/helper-function-name@^7.12.13": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/helper-function-name/-/helper-function-name-7.12.13.tgz#93ad656db3c3c2232559fd7b2c3dbdcbe0eb377a" - integrity sha512-TZvmPn0UOqmvi5G4vvw0qZTpVptGkB1GL61R6lKvrSdIxGm5Pky7Q3fpKiIkQCAtRCBUwB0PaThlx9vebCDSwA== - dependencies: - "@babel/helper-get-function-arity" "^7.12.13" - "@babel/template" "^7.12.13" - "@babel/types" "^7.12.13" - -"@babel/helper-get-function-arity@^7.12.13": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/helper-get-function-arity/-/helper-get-function-arity-7.12.13.tgz#bc63451d403a3b3082b97e1d8b3fe5bd4091e583" - integrity sha512-DjEVzQNz5LICkzN0REdpD5prGoidvbdYk1BVgRUOINaWJP2t6avB27X1guXK1kXNrX0WMfsrm1A/ZBthYuIMQg== - dependencies: - "@babel/types" "^7.12.13" - -"@babel/helper-hoist-variables@^7.13.0": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/helper-hoist-variables/-/helper-hoist-variables-7.13.0.tgz#5d5882e855b5c5eda91e0cadc26c6e7a2c8593d8" - integrity sha512-0kBzvXiIKfsCA0y6cFEIJf4OdzfpRuNk4+YTeHZpGGc666SATFKTz6sRncwFnQk7/ugJ4dSrCj6iJuvW4Qwr2g== - dependencies: - "@babel/traverse" "^7.13.0" - "@babel/types" "^7.13.0" - -"@babel/helper-member-expression-to-functions@^7.13.0": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.13.0.tgz#6aa4bb678e0f8c22f58cdb79451d30494461b091" - integrity sha512-yvRf8Ivk62JwisqV1rFRMxiSMDGnN6KH1/mDMmIrij4jztpQNRoHqqMG3U6apYbGRPJpgPalhva9Yd06HlUxJQ== - dependencies: - "@babel/types" "^7.13.0" - -"@babel/helper-module-imports@^7.12.1", "@babel/helper-module-imports@^7.12.13": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/helper-module-imports/-/helper-module-imports-7.12.13.tgz#ec67e4404f41750463e455cc3203f6a32e93fcb0" - integrity sha512-NGmfvRp9Rqxy0uHSSVP+SRIW1q31a7Ji10cLBcqSDUngGentY4FRiHOFZFE1CLU5eiL0oE8reH7Tg1y99TDM/g== - dependencies: - "@babel/types" "^7.12.13" - -"@babel/helper-module-transforms@^7.12.1", "@babel/helper-module-transforms@^7.13.0": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/helper-module-transforms/-/helper-module-transforms-7.13.0.tgz#42eb4bd8eea68bab46751212c357bfed8b40f6f1" - integrity sha512-Ls8/VBwH577+pw7Ku1QkUWIyRRNHpYlts7+qSqBBFCW3I8QteB9DxfcZ5YJpOwH6Ihe/wn8ch7fMGOP1OhEIvw== - dependencies: - "@babel/helper-module-imports" "^7.12.13" - "@babel/helper-replace-supers" "^7.13.0" - "@babel/helper-simple-access" "^7.12.13" - "@babel/helper-split-export-declaration" "^7.12.13" - "@babel/helper-validator-identifier" "^7.12.11" - "@babel/template" "^7.12.13" - "@babel/traverse" "^7.13.0" - "@babel/types" "^7.13.0" - lodash "^4.17.19" - -"@babel/helper-optimise-call-expression@^7.12.13": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.12.13.tgz#5c02d171b4c8615b1e7163f888c1c81c30a2aaea" - integrity sha512-BdWQhoVJkp6nVjB7nkFWcn43dkprYauqtk++Py2eaf/GRDFm5BxRqEIZCiHlZUGAVmtwKcsVL1dC68WmzeFmiA== - dependencies: - "@babel/types" "^7.12.13" - -"@babel/helper-plugin-utils@^7.0.0", "@babel/helper-plugin-utils@^7.10.4", "@babel/helper-plugin-utils@^7.12.13", "@babel/helper-plugin-utils@^7.13.0", "@babel/helper-plugin-utils@^7.8.0", "@babel/helper-plugin-utils@^7.8.3": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/helper-plugin-utils/-/helper-plugin-utils-7.13.0.tgz#806526ce125aed03373bc416a828321e3a6a33af" - integrity sha512-ZPafIPSwzUlAoWT8DKs1W2VyF2gOWthGd5NGFMsBcMMol+ZhK+EQY/e6V96poa6PA/Bh+C9plWN0hXO1uB8AfQ== - -"@babel/helper-remap-async-to-generator@^7.13.0": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.13.0.tgz#376a760d9f7b4b2077a9dd05aa9c3927cadb2209" - integrity sha512-pUQpFBE9JvC9lrQbpX0TmeNIy5s7GnZjna2lhhcHC7DzgBs6fWn722Y5cfwgrtrqc7NAJwMvOa0mKhq6XaE4jg== - dependencies: - "@babel/helper-annotate-as-pure" "^7.12.13" - "@babel/helper-wrap-function" "^7.13.0" - "@babel/types" "^7.13.0" - -"@babel/helper-replace-supers@^7.12.13", "@babel/helper-replace-supers@^7.13.0": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/helper-replace-supers/-/helper-replace-supers-7.13.0.tgz#6034b7b51943094cb41627848cb219cb02be1d24" - integrity sha512-Segd5me1+Pz+rmN/NFBOplMbZG3SqRJOBlY+mA0SxAv6rjj7zJqr1AVr3SfzUVTLCv7ZLU5FycOM/SBGuLPbZw== - dependencies: - "@babel/helper-member-expression-to-functions" "^7.13.0" - "@babel/helper-optimise-call-expression" "^7.12.13" - "@babel/traverse" "^7.13.0" - "@babel/types" "^7.13.0" - -"@babel/helper-simple-access@^7.12.13": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/helper-simple-access/-/helper-simple-access-7.12.13.tgz#8478bcc5cacf6aa1672b251c1d2dde5ccd61a6c4" - integrity sha512-0ski5dyYIHEfwpWGx5GPWhH35j342JaflmCeQmsPWcrOQDtCN6C1zKAVRFVbK53lPW2c9TsuLLSUDf0tIGJ5hA== - dependencies: - "@babel/types" "^7.12.13" - -"@babel/helper-skip-transparent-expression-wrappers@^7.12.1": - version "7.12.1" - resolved "https://registry.yarnpkg.com/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.12.1.tgz#462dc63a7e435ade8468385c63d2b84cce4b3cbf" - integrity sha512-Mf5AUuhG1/OCChOJ/HcADmvcHM42WJockombn8ATJG3OnyiSxBK/Mm5x78BQWvmtXZKHgbjdGL2kin/HOLlZGA== - dependencies: - "@babel/types" "^7.12.1" - -"@babel/helper-split-export-declaration@^7.12.13": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.12.13.tgz#e9430be00baf3e88b0e13e6f9d4eaf2136372b05" - integrity sha512-tCJDltF83htUtXx5NLcaDqRmknv652ZWCHyoTETf1CXYJdPC7nohZohjUgieXhv0hTJdRf2FjDueFehdNucpzg== - dependencies: - "@babel/types" "^7.12.13" - -"@babel/helper-validator-identifier@^7.12.11": - version "7.12.11" - resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.12.11.tgz#c9a1f021917dcb5ccf0d4e453e399022981fc9ed" - integrity sha512-np/lG3uARFybkoHokJUmf1QfEvRVCPbmQeUQpKow5cQ3xWrV9i3rUHodKDJPQfTVX61qKi+UdYk8kik84n7XOw== - -"@babel/helper-validator-option@^7.12.1", "@babel/helper-validator-option@^7.12.17": - version "7.12.17" - resolved "https://registry.yarnpkg.com/@babel/helper-validator-option/-/helper-validator-option-7.12.17.tgz#d1fbf012e1a79b7eebbfdc6d270baaf8d9eb9831" - integrity sha512-TopkMDmLzq8ngChwRlyjR6raKD6gMSae4JdYDB8bByKreQgG0RBTuKe9LRxW3wFtUnjxOPRKBDwEH6Mg5KeDfw== - -"@babel/helper-wrap-function@^7.13.0": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/helper-wrap-function/-/helper-wrap-function-7.13.0.tgz#bdb5c66fda8526ec235ab894ad53a1235c79fcc4" - integrity sha512-1UX9F7K3BS42fI6qd2A4BjKzgGjToscyZTdp1DjknHLCIvpgne6918io+aL5LXFcER/8QWiwpoY902pVEqgTXA== - dependencies: - "@babel/helper-function-name" "^7.12.13" - "@babel/template" "^7.12.13" - "@babel/traverse" "^7.13.0" - "@babel/types" "^7.13.0" - -"@babel/helpers@^7.12.1", "@babel/helpers@^7.13.10": - version "7.13.10" - resolved "https://registry.yarnpkg.com/@babel/helpers/-/helpers-7.13.10.tgz#fd8e2ba7488533cdeac45cc158e9ebca5e3c7df8" - integrity sha512-4VO883+MWPDUVRF3PhiLBUFHoX/bsLTGFpFK/HqvvfBZz2D57u9XzPVNFVBTc0PW/CWR9BXTOKt8NF4DInUHcQ== - dependencies: - "@babel/template" "^7.12.13" - "@babel/traverse" "^7.13.0" - "@babel/types" "^7.13.0" - -"@babel/highlight@^7.12.13": - version "7.13.10" - resolved "https://registry.yarnpkg.com/@babel/highlight/-/highlight-7.13.10.tgz#a8b2a66148f5b27d666b15d81774347a731d52d1" - integrity sha512-5aPpe5XQPzflQrFwL1/QoeHkP2MsA4JCntcXHRhEsdsfPVkvPi2w7Qix4iV7t5S/oC9OodGrggd8aco1g3SZFg== - dependencies: - "@babel/helper-validator-identifier" "^7.12.11" - chalk "^2.0.0" - js-tokens "^4.0.0" - -"@babel/parser@^7.10.4", "@babel/parser@^7.12.13", "@babel/parser@^7.12.3", "@babel/parser@^7.13.0", "@babel/parser@^7.13.10": - version "7.13.10" - resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.13.10.tgz#8f8f9bf7b3afa3eabd061f7a5bcdf4fec3c48409" - integrity sha512-0s7Mlrw9uTWkYua7xWr99Wpk2bnGa0ANleKfksYAES8LpWH4gW1OUr42vqKNf0us5UQNfru2wPqMqRITzq/SIQ== - -"@babel/plugin-proposal-async-generator-functions@^7.12.1": - version "7.13.8" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.13.8.tgz#87aacb574b3bc4b5603f6fe41458d72a5a2ec4b1" - integrity sha512-rPBnhj+WgoSmgq+4gQUtXx/vOcU+UYtjy1AA/aeD61Hwj410fwYyqfUcRP3lR8ucgliVJL/G7sXcNUecC75IXA== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/helper-remap-async-to-generator" "^7.13.0" - "@babel/plugin-syntax-async-generators" "^7.8.4" - -"@babel/plugin-proposal-class-properties@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-class-properties/-/plugin-proposal-class-properties-7.13.0.tgz#146376000b94efd001e57a40a88a525afaab9f37" - integrity sha512-KnTDjFNC1g+45ka0myZNvSBFLhNCLN+GeGYLDEA8Oq7MZ6yMgfLoIRh86GRT0FjtJhZw8JyUskP9uvj5pHM9Zg== - dependencies: - "@babel/helper-create-class-features-plugin" "^7.13.0" - "@babel/helper-plugin-utils" "^7.13.0" - -"@babel/plugin-proposal-dynamic-import@^7.12.1": - version "7.13.8" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-dynamic-import/-/plugin-proposal-dynamic-import-7.13.8.tgz#876a1f6966e1dec332e8c9451afda3bebcdf2e1d" - integrity sha512-ONWKj0H6+wIRCkZi9zSbZtE/r73uOhMVHh256ys0UzfM7I3d4n+spZNWjOnJv2gzopumP2Wxi186vI8N0Y2JyQ== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/plugin-syntax-dynamic-import" "^7.8.3" - -"@babel/plugin-proposal-export-namespace-from@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-export-namespace-from/-/plugin-proposal-export-namespace-from-7.12.13.tgz#393be47a4acd03fa2af6e3cde9b06e33de1b446d" - integrity sha512-INAgtFo4OnLN3Y/j0VwAgw3HDXcDtX+C/erMvWzuV9v71r7urb6iyMXu7eM9IgLr1ElLlOkaHjJ0SbCmdOQ3Iw== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - "@babel/plugin-syntax-export-namespace-from" "^7.8.3" - -"@babel/plugin-proposal-json-strings@^7.12.1": - version "7.13.8" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-json-strings/-/plugin-proposal-json-strings-7.13.8.tgz#bf1fb362547075afda3634ed31571c5901afef7b" - integrity sha512-w4zOPKUFPX1mgvTmL/fcEqy34hrQ1CRcGxdphBc6snDnnqJ47EZDIyop6IwXzAC8G916hsIuXB2ZMBCExC5k7Q== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/plugin-syntax-json-strings" "^7.8.3" - -"@babel/plugin-proposal-logical-assignment-operators@^7.12.1": - version "7.13.8" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-logical-assignment-operators/-/plugin-proposal-logical-assignment-operators-7.13.8.tgz#93fa78d63857c40ce3c8c3315220fd00bfbb4e1a" - integrity sha512-aul6znYB4N4HGweImqKn59Su9RS8lbUIqxtXTOcAGtNIDczoEFv+l1EhmX8rUBp3G1jMjKJm8m0jXVp63ZpS4A== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/plugin-syntax-logical-assignment-operators" "^7.10.4" - -"@babel/plugin-proposal-nullish-coalescing-operator@^7.12.1": - version "7.13.8" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-nullish-coalescing-operator/-/plugin-proposal-nullish-coalescing-operator-7.13.8.tgz#3730a31dafd3c10d8ccd10648ed80a2ac5472ef3" - integrity sha512-iePlDPBn//UhxExyS9KyeYU7RM9WScAG+D3Hhno0PLJebAEpDZMocbDe64eqynhNAnwz/vZoL/q/QB2T1OH39A== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/plugin-syntax-nullish-coalescing-operator" "^7.8.3" - -"@babel/plugin-proposal-numeric-separator@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-numeric-separator/-/plugin-proposal-numeric-separator-7.12.13.tgz#bd9da3188e787b5120b4f9d465a8261ce67ed1db" - integrity sha512-O1jFia9R8BUCl3ZGB7eitaAPu62TXJRHn7rh+ojNERCFyqRwJMTmhz+tJ+k0CwI6CLjX/ee4qW74FSqlq9I35w== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - "@babel/plugin-syntax-numeric-separator" "^7.10.4" - -"@babel/plugin-proposal-object-rest-spread@^7.12.1": - version "7.13.8" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.13.8.tgz#5d210a4d727d6ce3b18f9de82cc99a3964eed60a" - integrity sha512-DhB2EuB1Ih7S3/IRX5AFVgZ16k3EzfRbq97CxAVI1KSYcW+lexV8VZb7G7L8zuPVSdQMRn0kiBpf/Yzu9ZKH0g== - dependencies: - "@babel/compat-data" "^7.13.8" - "@babel/helper-compilation-targets" "^7.13.8" - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/plugin-syntax-object-rest-spread" "^7.8.3" - "@babel/plugin-transform-parameters" "^7.13.0" - -"@babel/plugin-proposal-optional-catch-binding@^7.12.1": - version "7.13.8" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.13.8.tgz#3ad6bd5901506ea996fc31bdcf3ccfa2bed71107" - integrity sha512-0wS/4DUF1CuTmGo+NiaHfHcVSeSLj5S3e6RivPTg/2k3wOv3jO35tZ6/ZWsQhQMvdgI7CwphjQa/ccarLymHVA== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/plugin-syntax-optional-catch-binding" "^7.8.3" - -"@babel/plugin-proposal-optional-chaining@^7.12.1": - version "7.13.8" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-optional-chaining/-/plugin-proposal-optional-chaining-7.13.8.tgz#e39df93efe7e7e621841babc197982e140e90756" - integrity sha512-hpbBwbTgd7Cz1QryvwJZRo1U0k1q8uyBmeXOSQUjdg/A2TASkhR/rz7AyqZ/kS8kbpsNA80rOYbxySBJAqmhhQ== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/helper-skip-transparent-expression-wrappers" "^7.12.1" - "@babel/plugin-syntax-optional-chaining" "^7.8.3" - -"@babel/plugin-proposal-private-methods@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-private-methods/-/plugin-proposal-private-methods-7.13.0.tgz#04bd4c6d40f6e6bbfa2f57e2d8094bad900ef787" - integrity sha512-MXyyKQd9inhx1kDYPkFRVOBXQ20ES8Pto3T7UZ92xj2mY0EVD8oAVzeyYuVfy/mxAdTSIayOvg+aVzcHV2bn6Q== - dependencies: - "@babel/helper-create-class-features-plugin" "^7.13.0" - "@babel/helper-plugin-utils" "^7.13.0" - -"@babel/plugin-proposal-unicode-property-regex@^7.12.1", "@babel/plugin-proposal-unicode-property-regex@^7.4.4": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.12.13.tgz#bebde51339be829c17aaaaced18641deb62b39ba" - integrity sha512-XyJmZidNfofEkqFV5VC/bLabGmO5QzenPO/YOfGuEbgU+2sSwMmio3YLb4WtBgcmmdwZHyVyv8on77IUjQ5Gvg== - dependencies: - "@babel/helper-create-regexp-features-plugin" "^7.12.13" - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-syntax-async-generators@^7.8.0", "@babel/plugin-syntax-async-generators@^7.8.4": - version "7.8.4" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz#a983fb1aeb2ec3f6ed042a210f640e90e786fe0d" - integrity sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw== - dependencies: - "@babel/helper-plugin-utils" "^7.8.0" - -"@babel/plugin-syntax-class-properties@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.13.tgz#b5c987274c4a3a82b89714796931a6b53544ae10" - integrity sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-syntax-dynamic-import@^7.8.0", "@babel/plugin-syntax-dynamic-import@^7.8.3": - version "7.8.3" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-dynamic-import/-/plugin-syntax-dynamic-import-7.8.3.tgz#62bf98b2da3cd21d626154fc96ee5b3cb68eacb3" - integrity sha512-5gdGbFon+PszYzqs83S3E5mpi7/y/8M9eC90MRTZfduQOYW76ig6SOSPNe41IG5LoP3FGBn2N0RjVDSQiS94kQ== - dependencies: - "@babel/helper-plugin-utils" "^7.8.0" - -"@babel/plugin-syntax-export-namespace-from@^7.8.3": - version "7.8.3" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-export-namespace-from/-/plugin-syntax-export-namespace-from-7.8.3.tgz#028964a9ba80dbc094c915c487ad7c4e7a66465a" - integrity sha512-MXf5laXo6c1IbEbegDmzGPwGNTsHZmEy6QGznu5Sh2UCWvueywb2ee+CCE4zQiZstxU9BMoQO9i6zUFSY0Kj0Q== - dependencies: - "@babel/helper-plugin-utils" "^7.8.3" - -"@babel/plugin-syntax-json-strings@^7.8.0", "@babel/plugin-syntax-json-strings@^7.8.3": - version "7.8.3" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz#01ca21b668cd8218c9e640cb6dd88c5412b2c96a" - integrity sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA== - dependencies: - "@babel/helper-plugin-utils" "^7.8.0" - -"@babel/plugin-syntax-logical-assignment-operators@^7.10.4": - version "7.10.4" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz#ca91ef46303530448b906652bac2e9fe9941f699" - integrity sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig== - dependencies: - "@babel/helper-plugin-utils" "^7.10.4" - -"@babel/plugin-syntax-nullish-coalescing-operator@^7.8.0", "@babel/plugin-syntax-nullish-coalescing-operator@^7.8.3": - version "7.8.3" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz#167ed70368886081f74b5c36c65a88c03b66d1a9" - integrity sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ== - dependencies: - "@babel/helper-plugin-utils" "^7.8.0" - -"@babel/plugin-syntax-numeric-separator@^7.10.4": - version "7.10.4" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz#b9b070b3e33570cd9fd07ba7fa91c0dd37b9af97" - integrity sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug== - dependencies: - "@babel/helper-plugin-utils" "^7.10.4" - -"@babel/plugin-syntax-object-rest-spread@^7.8.0", "@babel/plugin-syntax-object-rest-spread@^7.8.3": - version "7.8.3" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz#60e225edcbd98a640332a2e72dd3e66f1af55871" - integrity sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA== - dependencies: - "@babel/helper-plugin-utils" "^7.8.0" - -"@babel/plugin-syntax-optional-catch-binding@^7.8.0", "@babel/plugin-syntax-optional-catch-binding@^7.8.3": - version "7.8.3" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz#6111a265bcfb020eb9efd0fdfd7d26402b9ed6c1" - integrity sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q== - dependencies: - "@babel/helper-plugin-utils" "^7.8.0" - -"@babel/plugin-syntax-optional-chaining@^7.8.0", "@babel/plugin-syntax-optional-chaining@^7.8.3": - version "7.8.3" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz#4f69c2ab95167e0180cd5336613f8c5788f7d48a" - integrity sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg== - dependencies: - "@babel/helper-plugin-utils" "^7.8.0" - -"@babel/plugin-syntax-top-level-await@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.12.13.tgz#c5f0fa6e249f5b739727f923540cf7a806130178" - integrity sha512-A81F9pDwyS7yM//KwbCSDqy3Uj4NMIurtplxphWxoYtNPov7cJsDkAFNNyVlIZ3jwGycVsurZ+LtOA8gZ376iQ== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-arrow-functions@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.13.0.tgz#10a59bebad52d637a027afa692e8d5ceff5e3dae" - integrity sha512-96lgJagobeVmazXFaDrbmCLQxBysKu7U6Do3mLsx27gf5Dk85ezysrs2BZUpXD703U/Su1xTBDxxar2oa4jAGg== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - -"@babel/plugin-transform-async-to-generator@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.13.0.tgz#8e112bf6771b82bf1e974e5e26806c5c99aa516f" - integrity sha512-3j6E004Dx0K3eGmhxVJxwwI89CTJrce7lg3UrtFuDAVQ/2+SJ/h/aSFOeE6/n0WB1GsOffsJp6MnPQNQ8nmwhg== - dependencies: - "@babel/helper-module-imports" "^7.12.13" - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/helper-remap-async-to-generator" "^7.13.0" - -"@babel/plugin-transform-block-scoped-functions@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.12.13.tgz#a9bf1836f2a39b4eb6cf09967739de29ea4bf4c4" - integrity sha512-zNyFqbc3kI/fVpqwfqkg6RvBgFpC4J18aKKMmv7KdQ/1GgREapSJAykLMVNwfRGO3BtHj3YQZl8kxCXPcVMVeg== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-block-scoping@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.12.13.tgz#f36e55076d06f41dfd78557ea039c1b581642e61" - integrity sha512-Pxwe0iqWJX4fOOM2kEZeUuAxHMWb9nK+9oh5d11bsLoB0xMg+mkDpt0eYuDZB7ETrY9bbcVlKUGTOGWy7BHsMQ== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-classes@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-classes/-/plugin-transform-classes-7.13.0.tgz#0265155075c42918bf4d3a4053134176ad9b533b" - integrity sha512-9BtHCPUARyVH1oXGcSJD3YpsqRLROJx5ZNP6tN5vnk17N0SVf9WCtf8Nuh1CFmgByKKAIMstitKduoCmsaDK5g== - dependencies: - "@babel/helper-annotate-as-pure" "^7.12.13" - "@babel/helper-function-name" "^7.12.13" - "@babel/helper-optimise-call-expression" "^7.12.13" - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/helper-replace-supers" "^7.13.0" - "@babel/helper-split-export-declaration" "^7.12.13" - globals "^11.1.0" - -"@babel/plugin-transform-computed-properties@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.13.0.tgz#845c6e8b9bb55376b1fa0b92ef0bdc8ea06644ed" - integrity sha512-RRqTYTeZkZAz8WbieLTvKUEUxZlUTdmL5KGMyZj7FnMfLNKV4+r5549aORG/mgojRmFlQMJDUupwAMiF2Q7OUg== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - -"@babel/plugin-transform-destructuring@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.13.0.tgz#c5dce270014d4e1ebb1d806116694c12b7028963" - integrity sha512-zym5em7tePoNT9s964c0/KU3JPPnuq7VhIxPRefJ4/s82cD+q1mgKfuGRDMCPL0HTyKz4dISuQlCusfgCJ86HA== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - -"@babel/plugin-transform-dotall-regex@^7.12.1", "@babel/plugin-transform-dotall-regex@^7.4.4": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.12.13.tgz#3f1601cc29905bfcb67f53910f197aeafebb25ad" - integrity sha512-foDrozE65ZFdUC2OfgeOCrEPTxdB3yjqxpXh8CH+ipd9CHd4s/iq81kcUpyH8ACGNEPdFqbtzfgzbT/ZGlbDeQ== - dependencies: - "@babel/helper-create-regexp-features-plugin" "^7.12.13" - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-duplicate-keys@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.12.13.tgz#6f06b87a8b803fd928e54b81c258f0a0033904de" - integrity sha512-NfADJiiHdhLBW3pulJlJI2NB0t4cci4WTZ8FtdIuNc2+8pslXdPtRRAEWqUY+m9kNOk2eRYbTAOipAxlrOcwwQ== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-exponentiation-operator@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.12.13.tgz#4d52390b9a273e651e4aba6aee49ef40e80cd0a1" - integrity sha512-fbUelkM1apvqez/yYx1/oICVnGo2KM5s63mhGylrmXUxK/IAXSIf87QIxVfZldWf4QsOafY6vV3bX8aMHSvNrA== - dependencies: - "@babel/helper-builder-binary-assignment-operator-visitor" "^7.12.13" - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-for-of@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.13.0.tgz#c799f881a8091ac26b54867a845c3e97d2696062" - integrity sha512-IHKT00mwUVYE0zzbkDgNRP6SRzvfGCYsOxIRz8KsiaaHCcT9BWIkO+H9QRJseHBLOGBZkHUdHiqj6r0POsdytg== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - -"@babel/plugin-transform-function-name@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.12.13.tgz#bb024452f9aaed861d374c8e7a24252ce3a50051" - integrity sha512-6K7gZycG0cmIwwF7uMK/ZqeCikCGVBdyP2J5SKNCXO5EOHcqi+z7Jwf8AmyDNcBgxET8DrEtCt/mPKPyAzXyqQ== - dependencies: - "@babel/helper-function-name" "^7.12.13" - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-literals@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-literals/-/plugin-transform-literals-7.12.13.tgz#2ca45bafe4a820197cf315794a4d26560fe4bdb9" - integrity sha512-FW+WPjSR7hiUxMcKqyNjP05tQ2kmBCdpEpZHY1ARm96tGQCCBvXKnpjILtDplUnJ/eHZ0lALLM+d2lMFSpYJrQ== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-member-expression-literals@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.12.13.tgz#5ffa66cd59b9e191314c9f1f803b938e8c081e40" - integrity sha512-kxLkOsg8yir4YeEPHLuO2tXP9R/gTjpuTOjshqSpELUN3ZAg2jfDnKUvzzJxObun38sw3wm4Uu69sX/zA7iRvg== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-modules-amd@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.13.0.tgz#19f511d60e3d8753cc5a6d4e775d3a5184866cc3" - integrity sha512-EKy/E2NHhY/6Vw5d1k3rgoobftcNUmp9fGjb9XZwQLtTctsRBOTRO7RHHxfIky1ogMN5BxN7p9uMA3SzPfotMQ== - dependencies: - "@babel/helper-module-transforms" "^7.13.0" - "@babel/helper-plugin-utils" "^7.13.0" - babel-plugin-dynamic-import-node "^2.3.3" - -"@babel/plugin-transform-modules-commonjs@^7.12.1": - version "7.13.8" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.13.8.tgz#7b01ad7c2dcf2275b06fa1781e00d13d420b3e1b" - integrity sha512-9QiOx4MEGglfYZ4XOnU79OHr6vIWUakIj9b4mioN8eQIoEh+pf5p/zEB36JpDFWA12nNMiRf7bfoRvl9Rn79Bw== - dependencies: - "@babel/helper-module-transforms" "^7.13.0" - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/helper-simple-access" "^7.12.13" - babel-plugin-dynamic-import-node "^2.3.3" - -"@babel/plugin-transform-modules-systemjs@^7.12.1": - version "7.13.8" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.13.8.tgz#6d066ee2bff3c7b3d60bf28dec169ad993831ae3" - integrity sha512-hwqctPYjhM6cWvVIlOIe27jCIBgHCsdH2xCJVAYQm7V5yTMoilbVMi9f6wKg0rpQAOn6ZG4AOyvCqFF/hUh6+A== - dependencies: - "@babel/helper-hoist-variables" "^7.13.0" - "@babel/helper-module-transforms" "^7.13.0" - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/helper-validator-identifier" "^7.12.11" - babel-plugin-dynamic-import-node "^2.3.3" - -"@babel/plugin-transform-modules-umd@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.13.0.tgz#8a3d96a97d199705b9fd021580082af81c06e70b" - integrity sha512-D/ILzAh6uyvkWjKKyFE/W0FzWwasv6vPTSqPcjxFqn6QpX3u8DjRVliq4F2BamO2Wee/om06Vyy+vPkNrd4wxw== - dependencies: - "@babel/helper-module-transforms" "^7.13.0" - "@babel/helper-plugin-utils" "^7.13.0" - -"@babel/plugin-transform-named-capturing-groups-regex@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.12.13.tgz#2213725a5f5bbbe364b50c3ba5998c9599c5c9d9" - integrity sha512-Xsm8P2hr5hAxyYblrfACXpQKdQbx4m2df9/ZZSQ8MAhsadw06+jW7s9zsSw6he+mJZXRlVMyEnVktJo4zjk1WA== - dependencies: - "@babel/helper-create-regexp-features-plugin" "^7.12.13" - -"@babel/plugin-transform-new-target@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.12.13.tgz#e22d8c3af24b150dd528cbd6e685e799bf1c351c" - integrity sha512-/KY2hbLxrG5GTQ9zzZSc3xWiOy379pIETEhbtzwZcw9rvuaVV4Fqy7BYGYOWZnaoXIQYbbJ0ziXLa/sKcGCYEQ== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-object-super@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.12.13.tgz#b4416a2d63b8f7be314f3d349bd55a9c1b5171f7" - integrity sha512-JzYIcj3XtYspZDV8j9ulnoMPZZnF/Cj0LUxPOjR89BdBVx+zYJI9MdMIlUZjbXDX+6YVeS6I3e8op+qQ3BYBoQ== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - "@babel/helper-replace-supers" "^7.12.13" - -"@babel/plugin-transform-parameters@^7.12.1", "@babel/plugin-transform-parameters@^7.13.0": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.13.0.tgz#8fa7603e3097f9c0b7ca1a4821bc2fb52e9e5007" - integrity sha512-Jt8k/h/mIwE2JFEOb3lURoY5C85ETcYPnbuAJ96zRBzh1XHtQZfs62ChZ6EP22QlC8c7Xqr9q+e1SU5qttwwjw== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - -"@babel/plugin-transform-property-literals@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.12.13.tgz#4e6a9e37864d8f1b3bc0e2dce7bf8857db8b1a81" - integrity sha512-nqVigwVan+lR+g8Fj8Exl0UQX2kymtjcWfMOYM1vTYEKujeyv2SkMgazf2qNcK7l4SDiKyTA/nHCPqL4e2zo1A== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-regenerator@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.12.13.tgz#b628bcc9c85260ac1aeb05b45bde25210194a2f5" - integrity sha512-lxb2ZAvSLyJ2PEe47hoGWPmW22v7CtSl9jW8mingV4H2sEX/JOcrAj2nPuGWi56ERUm2bUpjKzONAuT6HCn2EA== - dependencies: - regenerator-transform "^0.14.2" - -"@babel/plugin-transform-reserved-words@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.12.13.tgz#7d9988d4f06e0fe697ea1d9803188aa18b472695" - integrity sha512-xhUPzDXxZN1QfiOy/I5tyye+TRz6lA7z6xaT4CLOjPRMVg1ldRf0LHw0TDBpYL4vG78556WuHdyO9oi5UmzZBg== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-runtime@7.12.1": - version "7.12.1" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.12.1.tgz#04b792057eb460389ff6a4198e377614ea1e7ba5" - integrity sha512-Ac/H6G9FEIkS2tXsZjL4RAdS3L3WHxci0usAnz7laPWUmFiGtj7tIASChqKZMHTSQTQY6xDbOq+V1/vIq3QrWg== - dependencies: - "@babel/helper-module-imports" "^7.12.1" - "@babel/helper-plugin-utils" "^7.10.4" - resolve "^1.8.1" - semver "^5.5.1" - -"@babel/plugin-transform-shorthand-properties@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.12.13.tgz#db755732b70c539d504c6390d9ce90fe64aff7ad" - integrity sha512-xpL49pqPnLtf0tVluuqvzWIgLEhuPpZzvs2yabUHSKRNlN7ScYU7aMlmavOeyXJZKgZKQRBlh8rHbKiJDraTSw== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-spread@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-spread/-/plugin-transform-spread-7.13.0.tgz#84887710e273c1815ace7ae459f6f42a5d31d5fd" - integrity sha512-V6vkiXijjzYeFmQTr3dBxPtZYLPcUfY34DebOU27jIl2M/Y8Egm52Hw82CSjjPqd54GTlJs5x+CR7HeNr24ckg== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - "@babel/helper-skip-transparent-expression-wrappers" "^7.12.1" - -"@babel/plugin-transform-sticky-regex@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.12.13.tgz#760ffd936face73f860ae646fb86ee82f3d06d1f" - integrity sha512-Jc3JSaaWT8+fr7GRvQP02fKDsYk4K/lYwWq38r/UGfaxo89ajud321NH28KRQ7xy1Ybc0VUE5Pz8psjNNDUglg== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-template-literals@^7.12.1": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.13.0.tgz#a36049127977ad94438dee7443598d1cefdf409d" - integrity sha512-d67umW6nlfmr1iehCcBv69eSUSySk1EsIS8aTDX4Xo9qajAh6mYtcl4kJrBkGXuxZPEgVr7RVfAvNW6YQkd4Mw== - dependencies: - "@babel/helper-plugin-utils" "^7.13.0" - -"@babel/plugin-transform-typeof-symbol@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.12.13.tgz#785dd67a1f2ea579d9c2be722de8c84cb85f5a7f" - integrity sha512-eKv/LmUJpMnu4npgfvs3LiHhJua5fo/CysENxa45YCQXZwKnGCQKAg87bvoqSW1fFT+HA32l03Qxsm8ouTY3ZQ== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-unicode-escapes@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-escapes/-/plugin-transform-unicode-escapes-7.12.13.tgz#840ced3b816d3b5127dd1d12dcedc5dead1a5e74" - integrity sha512-0bHEkdwJ/sN/ikBHfSmOXPypN/beiGqjo+o4/5K+vxEFNPRPdImhviPakMKG4x96l85emoa0Z6cDflsdBusZbw== - dependencies: - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/plugin-transform-unicode-regex@^7.12.1": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.12.13.tgz#b52521685804e155b1202e83fc188d34bb70f5ac" - integrity sha512-mDRzSNY7/zopwisPZ5kM9XKCfhchqIYwAKRERtEnhYscZB79VRekuRSoYbN0+KVe3y8+q1h6A4svXtP7N+UoCA== - dependencies: - "@babel/helper-create-regexp-features-plugin" "^7.12.13" - "@babel/helper-plugin-utils" "^7.12.13" - -"@babel/preset-env@7.12.1": - version "7.12.1" - resolved "https://registry.yarnpkg.com/@babel/preset-env/-/preset-env-7.12.1.tgz#9c7e5ca82a19efc865384bb4989148d2ee5d7ac2" - integrity sha512-H8kxXmtPaAGT7TyBvSSkoSTUK6RHh61So05SyEbpmr0MCZrsNYn7mGMzzeYoOUCdHzww61k8XBft2TaES+xPLg== - dependencies: - "@babel/compat-data" "^7.12.1" - "@babel/helper-compilation-targets" "^7.12.1" - "@babel/helper-module-imports" "^7.12.1" - "@babel/helper-plugin-utils" "^7.10.4" - "@babel/helper-validator-option" "^7.12.1" - "@babel/plugin-proposal-async-generator-functions" "^7.12.1" - "@babel/plugin-proposal-class-properties" "^7.12.1" - "@babel/plugin-proposal-dynamic-import" "^7.12.1" - "@babel/plugin-proposal-export-namespace-from" "^7.12.1" - "@babel/plugin-proposal-json-strings" "^7.12.1" - "@babel/plugin-proposal-logical-assignment-operators" "^7.12.1" - "@babel/plugin-proposal-nullish-coalescing-operator" "^7.12.1" - "@babel/plugin-proposal-numeric-separator" "^7.12.1" - "@babel/plugin-proposal-object-rest-spread" "^7.12.1" - "@babel/plugin-proposal-optional-catch-binding" "^7.12.1" - "@babel/plugin-proposal-optional-chaining" "^7.12.1" - "@babel/plugin-proposal-private-methods" "^7.12.1" - "@babel/plugin-proposal-unicode-property-regex" "^7.12.1" - "@babel/plugin-syntax-async-generators" "^7.8.0" - "@babel/plugin-syntax-class-properties" "^7.12.1" - "@babel/plugin-syntax-dynamic-import" "^7.8.0" - "@babel/plugin-syntax-export-namespace-from" "^7.8.3" - "@babel/plugin-syntax-json-strings" "^7.8.0" - "@babel/plugin-syntax-logical-assignment-operators" "^7.10.4" - "@babel/plugin-syntax-nullish-coalescing-operator" "^7.8.0" - "@babel/plugin-syntax-numeric-separator" "^7.10.4" - "@babel/plugin-syntax-object-rest-spread" "^7.8.0" - "@babel/plugin-syntax-optional-catch-binding" "^7.8.0" - "@babel/plugin-syntax-optional-chaining" "^7.8.0" - "@babel/plugin-syntax-top-level-await" "^7.12.1" - "@babel/plugin-transform-arrow-functions" "^7.12.1" - "@babel/plugin-transform-async-to-generator" "^7.12.1" - "@babel/plugin-transform-block-scoped-functions" "^7.12.1" - "@babel/plugin-transform-block-scoping" "^7.12.1" - "@babel/plugin-transform-classes" "^7.12.1" - "@babel/plugin-transform-computed-properties" "^7.12.1" - "@babel/plugin-transform-destructuring" "^7.12.1" - "@babel/plugin-transform-dotall-regex" "^7.12.1" - "@babel/plugin-transform-duplicate-keys" "^7.12.1" - "@babel/plugin-transform-exponentiation-operator" "^7.12.1" - "@babel/plugin-transform-for-of" "^7.12.1" - "@babel/plugin-transform-function-name" "^7.12.1" - "@babel/plugin-transform-literals" "^7.12.1" - "@babel/plugin-transform-member-expression-literals" "^7.12.1" - "@babel/plugin-transform-modules-amd" "^7.12.1" - "@babel/plugin-transform-modules-commonjs" "^7.12.1" - "@babel/plugin-transform-modules-systemjs" "^7.12.1" - "@babel/plugin-transform-modules-umd" "^7.12.1" - "@babel/plugin-transform-named-capturing-groups-regex" "^7.12.1" - "@babel/plugin-transform-new-target" "^7.12.1" - "@babel/plugin-transform-object-super" "^7.12.1" - "@babel/plugin-transform-parameters" "^7.12.1" - "@babel/plugin-transform-property-literals" "^7.12.1" - "@babel/plugin-transform-regenerator" "^7.12.1" - "@babel/plugin-transform-reserved-words" "^7.12.1" - "@babel/plugin-transform-shorthand-properties" "^7.12.1" - "@babel/plugin-transform-spread" "^7.12.1" - "@babel/plugin-transform-sticky-regex" "^7.12.1" - "@babel/plugin-transform-template-literals" "^7.12.1" - "@babel/plugin-transform-typeof-symbol" "^7.12.1" - "@babel/plugin-transform-unicode-escapes" "^7.12.1" - "@babel/plugin-transform-unicode-regex" "^7.12.1" - "@babel/preset-modules" "^0.1.3" - "@babel/types" "^7.12.1" - core-js-compat "^3.6.2" - semver "^5.5.0" - -"@babel/preset-modules@^0.1.3": - version "0.1.4" - resolved "https://registry.yarnpkg.com/@babel/preset-modules/-/preset-modules-0.1.4.tgz#362f2b68c662842970fdb5e254ffc8fc1c2e415e" - integrity sha512-J36NhwnfdzpmH41M1DrnkkgAqhZaqr/NBdPfQ677mLzlaXo+oDiv1deyCDtgAhz8p328otdob0Du7+xgHGZbKg== - dependencies: - "@babel/helper-plugin-utils" "^7.0.0" - "@babel/plugin-proposal-unicode-property-regex" "^7.4.4" - "@babel/plugin-transform-dotall-regex" "^7.4.4" - "@babel/types" "^7.4.4" - esutils "^2.0.2" - -"@babel/runtime@7.12.1": - version "7.12.1" - resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.12.1.tgz#b4116a6b6711d010b2dad3b7b6e43bf1b9954740" - integrity sha512-J5AIf3vPj3UwXaAzb5j1xM4WAQDX3EMgemF8rjCP3SoW09LfRKAXQKt6CoVYl230P6iWdRcBbnLDDdnqWxZSCA== - dependencies: - regenerator-runtime "^0.13.4" - -"@babel/runtime@^7.8.4": - version "7.13.10" - resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.13.10.tgz#47d42a57b6095f4468da440388fdbad8bebf0d7d" - integrity sha512-4QPkjJq6Ns3V/RgpEahRk+AGfL0eO6RHHtTWoNNr5mO49G6B5+X6d6THgWEAvTrznU5xYpbAlVKRYcsCgh/Akw== - dependencies: - regenerator-runtime "^0.13.4" - -"@babel/template@7.10.4": - version "7.10.4" - resolved "https://registry.yarnpkg.com/@babel/template/-/template-7.10.4.tgz#3251996c4200ebc71d1a8fc405fba940f36ba278" - integrity sha512-ZCjD27cGJFUB6nmCB1Enki3r+L5kJveX9pq1SvAUKoICy6CZ9yD8xO086YXdYhvNjBdnekm4ZnaP5yC8Cs/1tA== - dependencies: - "@babel/code-frame" "^7.10.4" - "@babel/parser" "^7.10.4" - "@babel/types" "^7.10.4" - -"@babel/template@^7.10.4", "@babel/template@^7.12.13": - version "7.12.13" - resolved "https://registry.yarnpkg.com/@babel/template/-/template-7.12.13.tgz#530265be8a2589dbb37523844c5bcb55947fb327" - integrity sha512-/7xxiGA57xMo/P2GVvdEumr8ONhFOhfgq2ihK3h1e6THqzTAkHbkXgB0xI9yeTfIUoH3+oAeHhqm/I43OTbbjA== - dependencies: - "@babel/code-frame" "^7.12.13" - "@babel/parser" "^7.12.13" - "@babel/types" "^7.12.13" - -"@babel/traverse@^7.12.1", "@babel/traverse@^7.13.0": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/traverse/-/traverse-7.13.0.tgz#6d95752475f86ee7ded06536de309a65fc8966cc" - integrity sha512-xys5xi5JEhzC3RzEmSGrs/b3pJW/o87SypZ+G/PhaE7uqVQNv/jlmVIBXuoh5atqQ434LfXV+sf23Oxj0bchJQ== - dependencies: - "@babel/code-frame" "^7.12.13" - "@babel/generator" "^7.13.0" - "@babel/helper-function-name" "^7.12.13" - "@babel/helper-split-export-declaration" "^7.12.13" - "@babel/parser" "^7.13.0" - "@babel/types" "^7.13.0" - debug "^4.1.0" - globals "^11.1.0" - lodash "^4.17.19" - -"@babel/types@^7.10.4", "@babel/types@^7.12.1", "@babel/types@^7.12.13", "@babel/types@^7.13.0", "@babel/types@^7.4.4", "@babel/types@^7.8.6": - version "7.13.0" - resolved "https://registry.yarnpkg.com/@babel/types/-/types-7.13.0.tgz#74424d2816f0171b4100f0ab34e9a374efdf7f80" - integrity sha512-hE+HE8rnG1Z6Wzo+MhaKE5lM5eMx71T4EHJgku2E3xIfaULhDcxiiRxUYgwX8qwP1BBSlag+TdGOt6JAidIZTA== - dependencies: - "@babel/helper-validator-identifier" "^7.12.11" - lodash "^4.17.19" - to-fast-properties "^2.0.0" - -"@electron/get@^1.0.1": - version "1.12.4" - resolved "https://registry.yarnpkg.com/@electron/get/-/get-1.12.4.tgz#a5971113fc1bf8fa12a8789dc20152a7359f06ab" - integrity sha512-6nr9DbJPUR9Xujw6zD3y+rS95TyItEVM0NVjt1EehY2vUWfIgPiIPVHxCvaTS0xr2B+DRxovYVKbuOWqC35kjg== - dependencies: - debug "^4.1.1" - env-paths "^2.2.0" - fs-extra "^8.1.0" - got "^9.6.0" - progress "^2.0.3" - semver "^6.2.0" - sumchecker "^3.0.1" - optionalDependencies: - global-agent "^2.0.2" - global-tunnel-ng "^2.7.1" - -"@istanbuljs/schema@^0.1.2": - version "0.1.3" - resolved "https://registry.yarnpkg.com/@istanbuljs/schema/-/schema-0.1.3.tgz#e45e384e4b8ec16bce2fd903af78450f6bf7ec98" - integrity sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA== - -"@jsdevtools/coverage-istanbul-loader@3.0.5": - version "3.0.5" - resolved "https://registry.yarnpkg.com/@jsdevtools/coverage-istanbul-loader/-/coverage-istanbul-loader-3.0.5.tgz#2a4bc65d0271df8d4435982db4af35d81754ee26" - integrity sha512-EUCPEkaRPvmHjWAAZkWMT7JDzpw7FKB00WTISaiXsbNOd5hCHg77XLA8sLYLFDo1zepYLo2w7GstN8YBqRXZfA== - dependencies: - convert-source-map "^1.7.0" - istanbul-lib-instrument "^4.0.3" - loader-utils "^2.0.0" - merge-source-map "^1.1.0" - schema-utils "^2.7.0" - -"@mdi/angular-material@^5.8.55": - version "5.9.55" - resolved "https://registry.yarnpkg.com/@mdi/angular-material/-/angular-material-5.9.55.tgz#e4fad2ae73d630f0853dc67e509ca50ea0cbef03" - integrity sha512-RAQlvSI90eJHW9o6uSN5KKH8SJ61ZQuXVCjLDpQWtKQX8aTRPTJkruZ1M2U0YT6AtxKY4hPW7qyHH7VTJdTXRA== - -"@ngtools/webpack@11.0.7": - version "11.0.7" - resolved "https://registry.yarnpkg.com/@ngtools/webpack/-/webpack-11.0.7.tgz#3ec8cd967fdf1dfc79e6b1469369148bc3a93bd1" - integrity sha512-OWGiiDc7s4T53BBCY8tLkLUjgw44HrixW8Wh8e4thFH1eIUM0NHe087s/B5hDNu72W/GqK4IoBbhNQ2wiCR7qQ== - dependencies: - "@angular-devkit/core" "11.0.7" - enhanced-resolve "5.3.1" - webpack-sources "2.0.1" - -"@nodelib/fs.scandir@2.1.4": - version "2.1.4" - resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.4.tgz#d4b3549a5db5de2683e0c1071ab4f140904bbf69" - integrity sha512-33g3pMJk3bg5nXbL/+CY6I2eJDzZAni49PfJnL5fghPTggPvBd/pFNSgJsdAgWptuFu7qq/ERvOYFlhvsLTCKA== - dependencies: - "@nodelib/fs.stat" "2.0.4" - run-parallel "^1.1.9" - -"@nodelib/fs.stat@2.0.4", "@nodelib/fs.stat@^2.0.2": - version "2.0.4" - resolved "https://registry.yarnpkg.com/@nodelib/fs.stat/-/fs.stat-2.0.4.tgz#a3f2dd61bab43b8db8fa108a121cfffe4c676655" - integrity sha512-IYlHJA0clt2+Vg7bccq+TzRdJvv19c2INqBSsoOLp1je7xjtr7J26+WXR72MCdvU9q1qTzIWDfhMf+DRvQJK4Q== - -"@nodelib/fs.walk@^1.2.3": - version "1.2.6" - resolved "https://registry.yarnpkg.com/@nodelib/fs.walk/-/fs.walk-1.2.6.tgz#cce9396b30aa5afe9e3756608f5831adcb53d063" - integrity sha512-8Broas6vTtW4GIXTAHDoE32hnN2M5ykgCpWGbuXHQ15vEMqr23pB76e/GZcYsZCHALv50ktd24qhEyKr6wBtow== - dependencies: - "@nodelib/fs.scandir" "2.1.4" - fastq "^1.6.0" - -"@npmcli/move-file@^1.0.1": - version "1.1.2" - resolved "https://registry.yarnpkg.com/@npmcli/move-file/-/move-file-1.1.2.tgz#1a82c3e372f7cae9253eb66d72543d6b8685c674" - integrity sha512-1SUf/Cg2GzGDyaf15aR9St9TWlb+XvbZXWpDx8YKs7MLzMH/BCeopv+y9vzrzgkfykCGuWOlSu3mZhj2+FQcrg== - dependencies: - mkdirp "^1.0.4" - rimraf "^3.0.2" - -"@schematics/angular@11.0.6": - version "11.0.6" - resolved "https://registry.yarnpkg.com/@schematics/angular/-/angular-11.0.6.tgz#5e52f8396e66138df0d6062130399fab830ee79e" - integrity sha512-XUcpOrlcp55PBHrgpIVx69lnhDY6ro35BSRmqNmjXik56qcOkfvdki8vvyW9EsWvu9/sfBSsVDdparlbVois7w== - dependencies: - "@angular-devkit/core" "11.0.6" - "@angular-devkit/schematics" "11.0.6" - jsonc-parser "2.3.1" - -"@schematics/update@0.1100.6": - version "0.1100.6" - resolved "https://registry.yarnpkg.com/@schematics/update/-/update-0.1100.6.tgz#8e76276a3daecfd698b39e7643bc21f3abb3a4d0" - integrity sha512-+B8n+k+zZ3VYOhjNBsLqzjp8O9ZdUWgdpf9L8XAA7mh/oPwufXpExyEc66uAS07imvUMmjz6i8E2eNWV/IjBJg== - dependencies: - "@angular-devkit/core" "11.0.6" - "@angular-devkit/schematics" "11.0.6" - "@yarnpkg/lockfile" "1.1.0" - ini "1.3.6" - npm-package-arg "^8.0.0" - pacote "9.5.12" - semver "7.3.2" - semver-intersect "1.4.0" - -"@sindresorhus/is@^0.14.0": - version "0.14.0" - resolved "https://registry.yarnpkg.com/@sindresorhus/is/-/is-0.14.0.tgz#9fb3a3cf3132328151f353de4632e01e52102bea" - integrity sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ== - -"@stomp/rx-stomp@^0.3.5": - version "0.3.5" - resolved "https://registry.yarnpkg.com/@stomp/rx-stomp/-/rx-stomp-0.3.5.tgz#3739f44f3f098ddfaf4f46d1f3b6541e3a1d4224" - integrity sha512-oEaMkq2IejzVKFg8EtSTe0xnyhPVWJxuQmnuTQStC3EAPcJgH9eQziJc0BR9tUvm6m962v4LPvyJKlzxOJhFWg== - dependencies: - "@stomp/stompjs" "^5.1.0 >=5.4.2" - angular2-uuid "^1.1.1" - -"@stomp/stompjs@^5.1.0 >=5.4.2": - version "5.4.4" - resolved "https://registry.yarnpkg.com/@stomp/stompjs/-/stompjs-5.4.4.tgz#f51d2edf9a00fac645dde3a494738d96ca17e5aa" - integrity sha512-RIzQ7MLRSJLUpTHcje1ZclnHH982amJSKC9bDxGO0wyu5OF9ROuuiLf7TxKxo1zUu7lGEYNedg9SEi87uMWDqg== - -"@szmarczak/http-timer@^1.1.2": - version "1.1.2" - resolved "https://registry.yarnpkg.com/@szmarczak/http-timer/-/http-timer-1.1.2.tgz#b1665e2c461a2cd92f4c1bbf50d5454de0d4b421" - integrity sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA== - dependencies: - defer-to-connect "^1.0.1" - -"@types/glob@^7.1.1": - version "7.1.3" - resolved "https://registry.yarnpkg.com/@types/glob/-/glob-7.1.3.tgz#e6ba80f36b7daad2c685acd9266382e68985c183" - integrity sha512-SEYeGAIQIQX8NN6LDKprLjbrd5dARM5EXsd8GI/A5l0apYI1fGMWgPHSe4ZKL4eozlAyI+doUE9XbYS4xCkQ1w== - dependencies: - "@types/minimatch" "*" - "@types/node" "*" - -"@types/jasmine@*": - version "3.6.6" - resolved "https://registry.yarnpkg.com/@types/jasmine/-/jasmine-3.6.6.tgz#4c0415f6198b2c643d895789f8c6d20436c1eddc" - integrity sha512-kgl+oYOLCBt41iba8cetp+QPOqDAaTJnHtVPCE7JzYmda4juglRBLX35opVcANc7TLA/Lp0DEnajbUNnyxGC+Q== - -"@types/jasmine@3.5.12": - version "3.5.12" - resolved "https://registry.yarnpkg.com/@types/jasmine/-/jasmine-3.5.12.tgz#5c378c1545cdc7cb339cff5578f854b6d1e0a17d" - integrity sha512-vJaQ58oceFao+NzpKNqLOWwHPsqA7YEhKv+mOXvYU4/qh+BfVWIxaBtL0Ck5iCS67yOkNwGkDCrzepnzIWF+7g== - -"@types/jasminewd2@2.0.8": - version "2.0.8" - resolved "https://registry.yarnpkg.com/@types/jasminewd2/-/jasminewd2-2.0.8.tgz#67afe5098d5ef2386073a7b7384b69a840dfe93b" - integrity sha512-d9p31r7Nxk0ZH0U39PTH0hiDlJ+qNVGjlt1ucOoTUptxb2v+Y5VMnsxfwN+i3hK4yQnqBi3FMmoMFcd1JHDxdg== - dependencies: - "@types/jasmine" "*" - -"@types/json-schema@^7.0.5", "@types/json-schema@^7.0.6": - version "7.0.7" - resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.7.tgz#98a993516c859eb0d5c4c8f098317a9ea68db9ad" - integrity sha512-cxWFQVseBm6O9Gbw1IWb8r6OS4OhSt3hPZLkFApLjM8TEXROBuQGLAH2i2gZpcXdLBIrpXuTDhH7Vbm1iXmNGA== - -"@types/minimatch@*": - version "3.0.3" - resolved "https://registry.yarnpkg.com/@types/minimatch/-/minimatch-3.0.3.tgz#3dca0e3f33b200fc7d1139c0cd96c1268cadfd9d" - integrity sha512-tHq6qdbT9U1IRSGf14CL0pUlULksvY9OZ+5eEgl1N7t+OA3tGvNpxJCzuKQlsNgCVwbAs670L1vcVQi8j9HjnA== - -"@types/node@*": - version "14.14.34" - resolved "https://registry.yarnpkg.com/@types/node/-/node-14.14.34.tgz#07935194fc049069a1c56c0c274265abeddf88da" - integrity sha512-dBPaxocOK6UVyvhbnpFIj2W+S+1cBTkHQbFQfeeJhoKFbzYcVUGHvddeWPSucKATb3F0+pgDq0i6ghEaZjsugA== - -"@types/node@12.12.21": - version "12.12.21" - resolved "https://registry.yarnpkg.com/@types/node/-/node-12.12.21.tgz#aa44a6363291c7037111c47e4661ad210aded23f" - integrity sha512-8sRGhbpU+ck1n0PGAUgVrWrWdjSW2aqNeyC15W88GRsMpSwzv6RJGlLhE7s2RhVSOdyDmxbqlWSeThq4/7xqlA== - -"@types/node@^12.0.12": - version "12.20.5" - resolved "https://registry.yarnpkg.com/@types/node/-/node-12.20.5.tgz#4ca82a766f05c359fd6c77505007e5a272f4bb9b" - integrity sha512-5Oy7tYZnu3a4pnJ//d4yVvOImExl4Vtwf0D40iKUlU+XlUsyV9iyFWyCFlwy489b72FMAik/EFwRkNLjjOdSPg== - -"@types/parse-json@^4.0.0": - version "4.0.0" - resolved "https://registry.yarnpkg.com/@types/parse-json/-/parse-json-4.0.0.tgz#2f8bb441434d163b35fb8ffdccd7138927ffb8c0" - integrity sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA== - -"@types/q@^0.0.32": - version "0.0.32" - resolved "https://registry.yarnpkg.com/@types/q/-/q-0.0.32.tgz#bd284e57c84f1325da702babfc82a5328190c0c5" - integrity sha1-vShOV8hPEyXacCur/IKlMoGQwMU= - -"@types/q@^1.5.1": - version "1.5.4" - resolved "https://registry.yarnpkg.com/@types/q/-/q-1.5.4.tgz#15925414e0ad2cd765bfef58842f7e26a7accb24" - integrity sha512-1HcDas8SEj4z1Wc696tH56G8OlRaH/sqZOynNNB+HF0WOeXPaxTtbYzJY2oEfiUxjSKjhCKr+MvR7dCHcEelug== - -"@types/selenium-webdriver@^3.0.0": - version "3.0.17" - resolved "https://registry.yarnpkg.com/@types/selenium-webdriver/-/selenium-webdriver-3.0.17.tgz#50bea0c3c2acc31c959c5b1e747798b3b3d06d4b" - integrity sha512-tGomyEuzSC1H28y2zlW6XPCaDaXFaD6soTdb4GNdmte2qfHtrKqhy0ZFs4r/1hpazCfEZqeTSRLvSasmEx89uw== - -"@types/source-list-map@*": - version "0.1.2" - resolved "https://registry.yarnpkg.com/@types/source-list-map/-/source-list-map-0.1.2.tgz#0078836063ffaf17412349bba364087e0ac02ec9" - integrity sha512-K5K+yml8LTo9bWJI/rECfIPrGgxdpeNbj+d53lwN4QjW1MCwlkhUms+gtdzigTeUyBr09+u8BwOIY3MXvHdcsA== - -"@types/webpack-sources@^0.1.5": - version "0.1.8" - resolved "https://registry.yarnpkg.com/@types/webpack-sources/-/webpack-sources-0.1.8.tgz#078d75410435993ec8a0a2855e88706f3f751f81" - integrity sha512-JHB2/xZlXOjzjBB6fMOpH1eQAfsrpqVVIbneE0Rok16WXwFaznaI5vfg75U5WgGJm7V9W1c4xeRQDjX/zwvghA== - dependencies: - "@types/node" "*" - "@types/source-list-map" "*" - source-map "^0.6.1" - -"@webassemblyjs/ast@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/ast/-/ast-1.9.0.tgz#bd850604b4042459a5a41cd7d338cbed695ed964" - integrity sha512-C6wW5L+b7ogSDVqymbkkvuW9kruN//YisMED04xzeBBqjHa2FYnmvOlS6Xj68xWQRgWvI9cIglsjFowH/RJyEA== - dependencies: - "@webassemblyjs/helper-module-context" "1.9.0" - "@webassemblyjs/helper-wasm-bytecode" "1.9.0" - "@webassemblyjs/wast-parser" "1.9.0" - -"@webassemblyjs/floating-point-hex-parser@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.9.0.tgz#3c3d3b271bddfc84deb00f71344438311d52ffb4" - integrity sha512-TG5qcFsS8QB4g4MhrxK5TqfdNe7Ey/7YL/xN+36rRjl/BlGE/NcBvJcqsRgCP6Z92mRE+7N50pRIi8SmKUbcQA== - -"@webassemblyjs/helper-api-error@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-api-error/-/helper-api-error-1.9.0.tgz#203f676e333b96c9da2eeab3ccef33c45928b6a2" - integrity sha512-NcMLjoFMXpsASZFxJ5h2HZRcEhDkvnNFOAKneP5RbKRzaWJN36NC4jqQHKwStIhGXu5mUWlUUk7ygdtrO8lbmw== - -"@webassemblyjs/helper-buffer@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-buffer/-/helper-buffer-1.9.0.tgz#a1442d269c5feb23fcbc9ef759dac3547f29de00" - integrity sha512-qZol43oqhq6yBPx7YM3m9Bv7WMV9Eevj6kMi6InKOuZxhw+q9hOkvq5e/PpKSiLfyetpaBnogSbNCfBwyB00CA== - -"@webassemblyjs/helper-code-frame@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-code-frame/-/helper-code-frame-1.9.0.tgz#647f8892cd2043a82ac0c8c5e75c36f1d9159f27" - integrity sha512-ERCYdJBkD9Vu4vtjUYe8LZruWuNIToYq/ME22igL+2vj2dQ2OOujIZr3MEFvfEaqKoVqpsFKAGsRdBSBjrIvZA== - dependencies: - "@webassemblyjs/wast-printer" "1.9.0" - -"@webassemblyjs/helper-fsm@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-fsm/-/helper-fsm-1.9.0.tgz#c05256b71244214671f4b08ec108ad63b70eddb8" - integrity sha512-OPRowhGbshCb5PxJ8LocpdX9Kl0uB4XsAjl6jH/dWKlk/mzsANvhwbiULsaiqT5GZGT9qinTICdj6PLuM5gslw== - -"@webassemblyjs/helper-module-context@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-module-context/-/helper-module-context-1.9.0.tgz#25d8884b76839871a08a6c6f806c3979ef712f07" - integrity sha512-MJCW8iGC08tMk2enck1aPW+BE5Cw8/7ph/VGZxwyvGbJwjktKkDK7vy7gAmMDx88D7mhDTCNKAW5tED+gZ0W8g== - dependencies: - "@webassemblyjs/ast" "1.9.0" - -"@webassemblyjs/helper-wasm-bytecode@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.9.0.tgz#4fed8beac9b8c14f8c58b70d124d549dd1fe5790" - integrity sha512-R7FStIzyNcd7xKxCZH5lE0Bqy+hGTwS3LJjuv1ZVxd9O7eHCedSdrId/hMOd20I+v8wDXEn+bjfKDLzTepoaUw== - -"@webassemblyjs/helper-wasm-section@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.9.0.tgz#5a4138d5a6292ba18b04c5ae49717e4167965346" - integrity sha512-XnMB8l3ek4tvrKUUku+IVaXNHz2YsJyOOmz+MMkZvh8h1uSJpSen6vYnw3IoQ7WwEuAhL8Efjms1ZWjqh2agvw== - dependencies: - "@webassemblyjs/ast" "1.9.0" - "@webassemblyjs/helper-buffer" "1.9.0" - "@webassemblyjs/helper-wasm-bytecode" "1.9.0" - "@webassemblyjs/wasm-gen" "1.9.0" - -"@webassemblyjs/ieee754@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/ieee754/-/ieee754-1.9.0.tgz#15c7a0fbaae83fb26143bbacf6d6df1702ad39e4" - integrity sha512-dcX8JuYU/gvymzIHc9DgxTzUUTLexWwt8uCTWP3otys596io0L5aW02Gb1RjYpx2+0Jus1h4ZFqjla7umFniTg== - dependencies: - "@xtuc/ieee754" "^1.2.0" - -"@webassemblyjs/leb128@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/leb128/-/leb128-1.9.0.tgz#f19ca0b76a6dc55623a09cffa769e838fa1e1c95" - integrity sha512-ENVzM5VwV1ojs9jam6vPys97B/S65YQtv/aanqnU7D8aSoHFX8GyhGg0CMfyKNIHBuAVjy3tlzd5QMMINa7wpw== - dependencies: - "@xtuc/long" "4.2.2" - -"@webassemblyjs/utf8@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/utf8/-/utf8-1.9.0.tgz#04d33b636f78e6a6813227e82402f7637b6229ab" - integrity sha512-GZbQlWtopBTP0u7cHrEx+73yZKrQoBMpwkGEIqlacljhXCkVM1kMQge/Mf+csMJAjEdSwhOyLAS0AoR3AG5P8w== - -"@webassemblyjs/wasm-edit@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-edit/-/wasm-edit-1.9.0.tgz#3fe6d79d3f0f922183aa86002c42dd256cfee9cf" - integrity sha512-FgHzBm80uwz5M8WKnMTn6j/sVbqilPdQXTWraSjBwFXSYGirpkSWE2R9Qvz9tNiTKQvoKILpCuTjBKzOIm0nxw== - dependencies: - "@webassemblyjs/ast" "1.9.0" - "@webassemblyjs/helper-buffer" "1.9.0" - "@webassemblyjs/helper-wasm-bytecode" "1.9.0" - "@webassemblyjs/helper-wasm-section" "1.9.0" - "@webassemblyjs/wasm-gen" "1.9.0" - "@webassemblyjs/wasm-opt" "1.9.0" - "@webassemblyjs/wasm-parser" "1.9.0" - "@webassemblyjs/wast-printer" "1.9.0" - -"@webassemblyjs/wasm-gen@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-gen/-/wasm-gen-1.9.0.tgz#50bc70ec68ded8e2763b01a1418bf43491a7a49c" - integrity sha512-cPE3o44YzOOHvlsb4+E9qSqjc9Qf9Na1OO/BHFy4OI91XDE14MjFN4lTMezzaIWdPqHnsTodGGNP+iRSYfGkjA== - dependencies: - "@webassemblyjs/ast" "1.9.0" - "@webassemblyjs/helper-wasm-bytecode" "1.9.0" - "@webassemblyjs/ieee754" "1.9.0" - "@webassemblyjs/leb128" "1.9.0" - "@webassemblyjs/utf8" "1.9.0" - -"@webassemblyjs/wasm-opt@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-opt/-/wasm-opt-1.9.0.tgz#2211181e5b31326443cc8112eb9f0b9028721a61" - integrity sha512-Qkjgm6Anhm+OMbIL0iokO7meajkzQD71ioelnfPEj6r4eOFuqm4YC3VBPqXjFyyNwowzbMD+hizmprP/Fwkl2A== - dependencies: - "@webassemblyjs/ast" "1.9.0" - "@webassemblyjs/helper-buffer" "1.9.0" - "@webassemblyjs/wasm-gen" "1.9.0" - "@webassemblyjs/wasm-parser" "1.9.0" - -"@webassemblyjs/wasm-parser@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-parser/-/wasm-parser-1.9.0.tgz#9d48e44826df4a6598294aa6c87469d642fff65e" - integrity sha512-9+wkMowR2AmdSWQzsPEjFU7njh8HTO5MqO8vjwEHuM+AMHioNqSBONRdr0NQQ3dVQrzp0s8lTcYqzUdb7YgELA== - dependencies: - "@webassemblyjs/ast" "1.9.0" - "@webassemblyjs/helper-api-error" "1.9.0" - "@webassemblyjs/helper-wasm-bytecode" "1.9.0" - "@webassemblyjs/ieee754" "1.9.0" - "@webassemblyjs/leb128" "1.9.0" - "@webassemblyjs/utf8" "1.9.0" - -"@webassemblyjs/wast-parser@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/wast-parser/-/wast-parser-1.9.0.tgz#3031115d79ac5bd261556cecc3fa90a3ef451914" - integrity sha512-qsqSAP3QQ3LyZjNC/0jBJ/ToSxfYJ8kYyuiGvtn/8MK89VrNEfwj7BPQzJVHi0jGTRK2dGdJ5PRqhtjzoww+bw== - dependencies: - "@webassemblyjs/ast" "1.9.0" - "@webassemblyjs/floating-point-hex-parser" "1.9.0" - "@webassemblyjs/helper-api-error" "1.9.0" - "@webassemblyjs/helper-code-frame" "1.9.0" - "@webassemblyjs/helper-fsm" "1.9.0" - "@xtuc/long" "4.2.2" - -"@webassemblyjs/wast-printer@1.9.0": - version "1.9.0" - resolved "https://registry.yarnpkg.com/@webassemblyjs/wast-printer/-/wast-printer-1.9.0.tgz#4935d54c85fef637b00ce9f52377451d00d47899" - integrity sha512-2J0nE95rHXHyQ24cWjMKJ1tqB/ds8z/cyeOZxJhcb+rW+SQASVjuznUSmdz5GpVJTzU8JkhYut0D3siFDD6wsA== - dependencies: - "@webassemblyjs/ast" "1.9.0" - "@webassemblyjs/wast-parser" "1.9.0" - "@xtuc/long" "4.2.2" - -"@xtuc/ieee754@^1.2.0": - version "1.2.0" - resolved "https://registry.yarnpkg.com/@xtuc/ieee754/-/ieee754-1.2.0.tgz#eef014a3145ae477a1cbc00cd1e552336dceb790" - integrity sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA== - -"@xtuc/long@4.2.2": - version "4.2.2" - resolved "https://registry.yarnpkg.com/@xtuc/long/-/long-4.2.2.tgz#d291c6a4e97989b5c61d9acf396ae4fe133a718d" - integrity sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ== - -"@yarnpkg/lockfile@1.1.0": - version "1.1.0" - resolved "https://registry.yarnpkg.com/@yarnpkg/lockfile/-/lockfile-1.1.0.tgz#e77a97fbd345b76d83245edcd17d393b1b41fb31" - integrity sha512-GpSwvyXOcOOlV70vbnzjj4fW5xW/FdUF6nQEt1ENy7m4ZCczi1+/buVUPAqmGfqznsORNFzUMjctTIp8a9tuCQ== - -JSONStream@^1.3.4: - version "1.3.5" - resolved "https://registry.yarnpkg.com/JSONStream/-/JSONStream-1.3.5.tgz#3208c1f08d3a4d99261ab64f92302bc15e111ca0" - integrity sha512-E+iruNOY8VV9s4JEbe1aNEm6MiszPRr/UfcHMz0TQh1BXSxHK+ASV1R6W4HpjBhSeS+54PIsAMCBmwD06LLsqQ== - dependencies: - jsonparse "^1.2.0" - through ">=2.2.7 <3" - -abab@^2.0.5: - version "2.0.5" - resolved "https://registry.yarnpkg.com/abab/-/abab-2.0.5.tgz#c0b678fb32d60fc1219c784d6a826fe385aeb79a" - integrity sha512-9IK9EadsbHo6jLWIpxpR6pL0sazTXV6+SQv25ZB+F7Bj9mJNaOc4nCRabwd5M/JwmUa8idz6Eci6eKfJryPs6Q== - -accepts@~1.3.4, accepts@~1.3.5, accepts@~1.3.7: - version "1.3.7" - resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.7.tgz#531bc726517a3b2b41f850021c6cc15eaab507cd" - integrity sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA== - dependencies: - mime-types "~2.1.24" - negotiator "0.6.2" - -acorn@^6.4.1: - version "6.4.2" - resolved "https://registry.yarnpkg.com/acorn/-/acorn-6.4.2.tgz#35866fd710528e92de10cf06016498e47e39e1e6" - integrity sha512-XtGIhXwF8YM8bJhGxG5kXgjkEuNGLTkoYqVE+KMR+aspr4KGYmKYg7yUe3KghyQ9yheNwLnjmzh/7+gfDBmHCQ== - -adjust-sourcemap-loader@3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/adjust-sourcemap-loader/-/adjust-sourcemap-loader-3.0.0.tgz#5ae12fb5b7b1c585e80bbb5a63ec163a1a45e61e" - integrity sha512-YBrGyT2/uVQ/c6Rr+t6ZJXniY03YtHGMJQYal368burRGYKqhx9qGTWqcBU5s1CwYY9E/ri63RYyG1IacMZtqw== - dependencies: - loader-utils "^2.0.0" - regex-parser "^2.2.11" - -adm-zip@^0.4.9: - version "0.4.16" - resolved "https://registry.yarnpkg.com/adm-zip/-/adm-zip-0.4.16.tgz#cf4c508fdffab02c269cbc7f471a875f05570365" - integrity sha512-TFi4HBKSGfIKsK5YCkKaaFG2m4PEDyViZmEwof3MTIgzimHLto6muaHVpbrljdIvIrFZzEq/p4nafOeLcYegrg== - -after@0.8.2: - version "0.8.2" - resolved "https://registry.yarnpkg.com/after/-/after-0.8.2.tgz#fedb394f9f0e02aa9768e702bda23b505fae7e1f" - integrity sha1-/ts5T58OAqqXaOcCvaI7UF+ufh8= - -ag-grid-angular@24.1.0: - version "24.1.0" - resolved "https://registry.yarnpkg.com/ag-grid-angular/-/ag-grid-angular-24.1.0.tgz#98c238003bf0dd46d43a2fd7d2f1aec1eeeafbac" - integrity sha512-rCcRWq6iC8yvRBk99qANSiiZGsUDo9E3PcFVDd7Xs2thRuJ/yrVhCSZEkCG/SlXHX2ZaMxzIxI+2ChPV2MJSEQ== - dependencies: - tslib "^1.10.0" - -ag-grid-community@23.2.1: - version "23.2.1" - resolved "https://registry.yarnpkg.com/ag-grid-community/-/ag-grid-community-23.2.1.tgz#0cf2beb7ba2fcf66a4865e45faabdbea173d59b6" - integrity sha512-WHfDV7mFsd6eQJ5yOu26nnpnW7Gd02zcp9LHHfKQky8lfgTEo382qcja7MibHzB2f5vOCBdQnOPsWrbBNKRzhQ== - -agent-base@4, agent-base@^4.3.0: - version "4.3.0" - resolved "https://registry.yarnpkg.com/agent-base/-/agent-base-4.3.0.tgz#8165f01c436009bccad0b1d122f05ed770efc6ee" - integrity sha512-salcGninV0nPrwpGNn4VTXBb1SOuXQBiqbrNXoeizJsHrsL6ERFM2Ne3JUSBWRE6aeNJI2ROP/WEEIDUiDe3cg== - dependencies: - es6-promisify "^5.0.0" - -agent-base@~4.2.1: - version "4.2.1" - resolved "https://registry.yarnpkg.com/agent-base/-/agent-base-4.2.1.tgz#d89e5999f797875674c07d87f260fc41e83e8ca9" - integrity sha512-JVwXMr9nHYTUXsBFKUqhJwvlcYU/blreOEUkhNR2eXZIvwd+c+o5V4MgDPKWnMS/56awN3TRzIP+KoPn+roQtg== - dependencies: - es6-promisify "^5.0.0" - -agentkeepalive@^3.4.1: - version "3.5.2" - resolved "https://registry.yarnpkg.com/agentkeepalive/-/agentkeepalive-3.5.2.tgz#a113924dd3fa24a0bc3b78108c450c2abee00f67" - integrity sha512-e0L/HNe6qkQ7H19kTlRRqUibEAwDK5AFk6y3PtMsuut2VAH6+Q4xZml1tNDJD7kSAyqmbG/K08K5WEJYtUrSlQ== - dependencies: - humanize-ms "^1.2.1" - -aggregate-error@^3.0.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/aggregate-error/-/aggregate-error-3.1.0.tgz#92670ff50f5359bdb7a3e0d40d0ec30c5737687a" - integrity sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA== - dependencies: - clean-stack "^2.0.0" - indent-string "^4.0.0" - -ajv-errors@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/ajv-errors/-/ajv-errors-1.0.1.tgz#f35986aceb91afadec4102fbd85014950cefa64d" - integrity sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ== - -ajv-keywords@^3.1.0, ajv-keywords@^3.4.1, ajv-keywords@^3.5.2: - version "3.5.2" - resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-3.5.2.tgz#31f29da5ab6e00d1c2d329acf7b5929614d5014d" - integrity sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ== - -ajv@6.12.6, ajv@^6.1.0, ajv@^6.10.2, ajv@^6.12.3, ajv@^6.12.4, ajv@^6.12.5: - version "6.12.6" - resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4" - integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g== - dependencies: - fast-deep-equal "^3.1.1" - fast-json-stable-stringify "^2.0.0" - json-schema-traverse "^0.4.1" - uri-js "^4.2.2" - -alphanum-sort@^1.0.0: - version "1.0.2" - resolved "https://registry.yarnpkg.com/alphanum-sort/-/alphanum-sort-1.0.2.tgz#97a1119649b211ad33691d9f9f486a8ec9fbe0a3" - integrity sha1-l6ERlkmyEa0zaR2fn0hqjsn74KM= - -angular2-uuid@^1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/angular2-uuid/-/angular2-uuid-1.1.1.tgz#72f03cd532b7f40032eb1ecfb9f8457384be956e" - integrity sha1-cvA81TK39AAy6x7PufhFc4S+lW4= - -ansi-colors@4.1.1: - version "4.1.1" - resolved "https://registry.yarnpkg.com/ansi-colors/-/ansi-colors-4.1.1.tgz#cbb9ae256bf750af1eab344f229aa27fe94ba348" - integrity sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA== - -ansi-colors@^3.0.0: - version "3.2.4" - resolved "https://registry.yarnpkg.com/ansi-colors/-/ansi-colors-3.2.4.tgz#e3a3da4bfbae6c86a9c285625de124a234026fbf" - integrity sha512-hHUXGagefjN2iRrID63xckIvotOXOojhQKWIPUZ4mNUZ9nLZW+7FMNoE1lOkEhNWYsx/7ysGIuJYCiMAA9FnrA== - -ansi-escapes@^4.2.1: - version "4.3.1" - resolved "https://registry.yarnpkg.com/ansi-escapes/-/ansi-escapes-4.3.1.tgz#a5c47cc43181f1f38ffd7076837700d395522a61" - integrity sha512-JWF7ocqNrp8u9oqpgV+wH5ftbt+cfvv+PTjOvKLT3AdYly/LmORARfEVT1iyjwN+4MqE5UmVKoAdIBqeoCHgLA== - dependencies: - type-fest "^0.11.0" - -ansi-html@0.0.7: - version "0.0.7" - resolved "https://registry.yarnpkg.com/ansi-html/-/ansi-html-0.0.7.tgz#813584021962a9e9e6fd039f940d12f56ca7859e" - integrity sha1-gTWEAhliqenm/QOflA0S9WynhZ4= - -ansi-regex@^2.0.0: - version "2.1.1" - resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-2.1.1.tgz#c3b33ab5ee360d86e0e628f0468ae7ef27d654df" - integrity sha1-w7M6te42DYbg5ijwRorn7yfWVN8= - -ansi-regex@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-4.1.0.tgz#8b9f8f08cf1acb843756a839ca8c7e3168c51997" - integrity sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg== - -ansi-regex@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.0.tgz#388539f55179bf39339c81af30a654d69f87cb75" - integrity sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg== - -ansi-styles@^2.2.1: - version "2.2.1" - resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-2.2.1.tgz#b432dd3358b634cf75e1e4664368240533c1ddbe" - integrity sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4= - -ansi-styles@^3.2.0, ansi-styles@^3.2.1: - version "3.2.1" - resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz#41fbb20243e50b12be0f04b8dedbf07520ce841d" - integrity sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA== - dependencies: - color-convert "^1.9.0" - -ansi-styles@^4.0.0, ansi-styles@^4.1.0: - version "4.3.0" - resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937" - integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg== - dependencies: - color-convert "^2.0.1" - -anymatch@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-2.0.0.tgz#bcb24b4f37934d9aa7ac17b4adaf89e7c76ef2eb" - integrity sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw== - dependencies: - micromatch "^3.1.4" - normalize-path "^2.1.1" - -anymatch@~3.1.1: - version "3.1.1" - resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-3.1.1.tgz#c55ecf02185e2469259399310c173ce31233b142" - integrity sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg== - dependencies: - normalize-path "^3.0.0" - picomatch "^2.0.4" - -app-root-path@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/app-root-path/-/app-root-path-3.0.0.tgz#210b6f43873227e18a4b810a032283311555d5ad" - integrity sha512-qMcx+Gy2UZynHjOHOIXPNvpf+9cjvk3cWrBBK7zg4gH9+clobJRb9NGzcT7mQTcV/6Gm/1WelUtqxVXnNlrwcw== - -aproba@^1.1.1: - version "1.2.0" - resolved "https://registry.yarnpkg.com/aproba/-/aproba-1.2.0.tgz#6802e6264efd18c790a1b0d517f0f2627bf2c94a" - integrity sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw== - -arg@^4.1.0: - version "4.1.3" - resolved "https://registry.yarnpkg.com/arg/-/arg-4.1.3.tgz#269fc7ad5b8e42cb63c896d5666017261c144089" - integrity sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA== - -argparse@^1.0.7: - version "1.0.10" - resolved "https://registry.yarnpkg.com/argparse/-/argparse-1.0.10.tgz#bcd6791ea5ae09725e17e5ad988134cd40b3d911" - integrity sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg== - dependencies: - sprintf-js "~1.0.2" - -aria-query@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/aria-query/-/aria-query-3.0.0.tgz#65b3fcc1ca1155a8c9ae64d6eee297f15d5133cc" - integrity sha1-ZbP8wcoRVajJrmTW7uKX8V1RM8w= - dependencies: - ast-types-flow "0.0.7" - commander "^2.11.0" - -arity-n@^1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/arity-n/-/arity-n-1.0.4.tgz#d9e76b11733e08569c0847ae7b39b2860b30b745" - integrity sha1-2edrEXM+CFacCEeuezmyhgswt0U= - -arr-diff@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/arr-diff/-/arr-diff-4.0.0.tgz#d6461074febfec71e7e15235761a329a5dc7c520" - integrity sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA= - -arr-flatten@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/arr-flatten/-/arr-flatten-1.1.0.tgz#36048bbff4e7b47e136644316c99669ea5ae91f1" - integrity sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg== - -arr-union@^3.1.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/arr-union/-/arr-union-3.1.0.tgz#e39b09aea9def866a8f206e288af63919bae39c4" - integrity sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ= - -array-flatten@1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2" - integrity sha1-ml9pkFGx5wczKPKgCJaLZOopVdI= - -array-flatten@^2.1.0: - version "2.1.2" - resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-2.1.2.tgz#24ef80a28c1a893617e2149b0c6d0d788293b099" - integrity sha512-hNfzcOV8W4NdualtqBFPyVO+54DSJuZGY9qT4pRroB6S9e3iiido2ISIC5h9R2sPJ8H3FHCIiEnsv1lPXO3KtQ== - -array-union@^1.0.1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/array-union/-/array-union-1.0.2.tgz#9a34410e4f4e3da23dea375be5be70f24778ec39" - integrity sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk= - dependencies: - array-uniq "^1.0.1" - -array-union@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/array-union/-/array-union-2.1.0.tgz#b798420adbeb1de828d84acd8a2e23d3efe85e8d" - integrity sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw== - -array-uniq@^1.0.1: - version "1.0.3" - resolved "https://registry.yarnpkg.com/array-uniq/-/array-uniq-1.0.3.tgz#af6ac877a25cc7f74e058894753858dfdb24fdb6" - integrity sha1-r2rId6Jcx/dOBYiUdThY39sk/bY= - -array-unique@^0.3.2: - version "0.3.2" - resolved "https://registry.yarnpkg.com/array-unique/-/array-unique-0.3.2.tgz#a894b75d4bc4f6cd679ef3244a9fd8f46ae2d428" - integrity sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg= - -arraybuffer.slice@~0.0.7: - version "0.0.7" - resolved "https://registry.yarnpkg.com/arraybuffer.slice/-/arraybuffer.slice-0.0.7.tgz#3bbc4275dd584cc1b10809b89d4e8b63a69e7675" - integrity sha512-wGUIVQXuehL5TCqQun8OW81jGzAWycqzFF8lFp+GOM5BXLYj3bKNsYC4daB7n6XjCqxQA/qgTJ+8ANR3acjrog== - -arrify@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/arrify/-/arrify-1.0.1.tgz#898508da2226f380df904728456849c1501a4b0d" - integrity sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0= - -asn1.js@^5.2.0: - version "5.4.1" - resolved "https://registry.yarnpkg.com/asn1.js/-/asn1.js-5.4.1.tgz#11a980b84ebb91781ce35b0fdc2ee294e3783f07" - integrity sha512-+I//4cYPccV8LdmBLiX8CYvf9Sp3vQsrqu2QNXRcrbiWvcx/UdlFiqUJJzxRQxgsZmvhXhn4cSKeSmoFjVdupA== - dependencies: - bn.js "^4.0.0" - inherits "^2.0.1" - minimalistic-assert "^1.0.0" - safer-buffer "^2.1.0" - -asn1@~0.2.3: - version "0.2.4" - resolved "https://registry.yarnpkg.com/asn1/-/asn1-0.2.4.tgz#8d2475dfab553bb33e77b54e59e880bb8ce23136" - integrity sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg== - dependencies: - safer-buffer "~2.1.0" - -assert-plus@1.0.0, assert-plus@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/assert-plus/-/assert-plus-1.0.0.tgz#f12e0f3c5d77b0b1cdd9146942e4e96c1e4dd525" - integrity sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU= - -assert@^1.1.1: - version "1.5.0" - resolved "https://registry.yarnpkg.com/assert/-/assert-1.5.0.tgz#55c109aaf6e0aefdb3dc4b71240c70bf574b18eb" - integrity sha512-EDsgawzwoun2CZkCgtxJbv392v4nbk9XDD06zI+kQYoBM/3RBWLlEyJARDOmhAAosBjWACEkKL6S+lIZtcAubA== - dependencies: - object-assign "^4.1.1" - util "0.10.3" - -assign-symbols@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/assign-symbols/-/assign-symbols-1.0.0.tgz#59667f41fadd4f20ccbc2bb96b8d4f7f78ec0367" - integrity sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c= - -ast-types-flow@0.0.7: - version "0.0.7" - resolved "https://registry.yarnpkg.com/ast-types-flow/-/ast-types-flow-0.0.7.tgz#f70b735c6bca1a5c9c22d982c3e39e7feba3bdad" - integrity sha1-9wtzXGvKGlycItmCw+Oef+ujva0= - -async-each@^1.0.1: - version "1.0.3" - resolved "https://registry.yarnpkg.com/async-each/-/async-each-1.0.3.tgz#b727dbf87d7651602f06f4d4ac387f47d91b0cbf" - integrity sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ== - -async-limiter@~1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/async-limiter/-/async-limiter-1.0.1.tgz#dd379e94f0db8310b08291f9d64c3209766617fd" - integrity sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ== - -async@^2.6.2: - version "2.6.3" - resolved "https://registry.yarnpkg.com/async/-/async-2.6.3.tgz#d72625e2344a3656e3a3ad4fa749fa83299d82ff" - integrity sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg== - dependencies: - lodash "^4.17.14" - -asynckit@^0.4.0: - version "0.4.0" - resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79" - integrity sha1-x57Zf380y48robyXkLzDZkdLS3k= - -atob@^2.1.2: - version "2.1.2" - resolved "https://registry.yarnpkg.com/atob/-/atob-2.1.2.tgz#6d9517eb9e030d2436666651e86bd9f6f13533c9" - integrity sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg== - -autoprefixer@9.8.6: - version "9.8.6" - resolved "https://registry.yarnpkg.com/autoprefixer/-/autoprefixer-9.8.6.tgz#3b73594ca1bf9266320c5acf1588d74dea74210f" - integrity sha512-XrvP4VVHdRBCdX1S3WXVD8+RyG9qeb1D5Sn1DeLiG2xfSpzellk5k54xbUERJ3M5DggQxes39UGOTP8CFrEGbg== - dependencies: - browserslist "^4.12.0" - caniuse-lite "^1.0.30001109" - colorette "^1.2.1" - normalize-range "^0.1.2" - num2fraction "^1.2.2" - postcss "^7.0.32" - postcss-value-parser "^4.1.0" - -aws-sign2@~0.7.0: - version "0.7.0" - resolved "https://registry.yarnpkg.com/aws-sign2/-/aws-sign2-0.7.0.tgz#b46e890934a9591f2d2f6f86d7e6a9f1b3fe76a8" - integrity sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg= - -aws4@^1.8.0: - version "1.11.0" - resolved "https://registry.yarnpkg.com/aws4/-/aws4-1.11.0.tgz#d61f46d83b2519250e2784daf5b09479a8b41c59" - integrity sha512-xh1Rl34h6Fi1DC2WWKfxUTVqRsNnr6LsKz2+hfwDxQJWmrx8+c7ylaqBMcHfl1U1r2dsifOvKX3LQuLNZ+XSvA== - -axobject-query@2.0.2: - version "2.0.2" - resolved "https://registry.yarnpkg.com/axobject-query/-/axobject-query-2.0.2.tgz#ea187abe5b9002b377f925d8bf7d1c561adf38f9" - integrity sha512-MCeek8ZH7hKyO1rWUbKNQBbl4l2eY0ntk7OGi+q0RlafrCnfPxC06WZA+uebCfmYp4mNU9jRBP1AhGyf8+W3ww== - dependencies: - ast-types-flow "0.0.7" - -babel-loader@8.1.0: - version "8.1.0" - resolved "https://registry.yarnpkg.com/babel-loader/-/babel-loader-8.1.0.tgz#c611d5112bd5209abe8b9fa84c3e4da25275f1c3" - integrity sha512-7q7nC1tYOrqvUrN3LQK4GwSk/TQorZSOlO9C+RZDZpODgyN4ZlCqE5q9cDsyWOliN+aU9B4JX01xK9eJXowJLw== - dependencies: - find-cache-dir "^2.1.0" - loader-utils "^1.4.0" - mkdirp "^0.5.3" - pify "^4.0.1" - schema-utils "^2.6.5" - -babel-plugin-dynamic-import-node@^2.3.3: - version "2.3.3" - resolved "https://registry.yarnpkg.com/babel-plugin-dynamic-import-node/-/babel-plugin-dynamic-import-node-2.3.3.tgz#84fda19c976ec5c6defef57f9427b3def66e17a3" - integrity sha512-jZVI+s9Zg3IqA/kdi0i6UDCybUI3aSBLnglhYbSSjKlV7yF1F/5LWv8MakQmvYpnbJDS6fcBL2KzHSxNCMtWSQ== - dependencies: - object.assign "^4.1.0" - -backo2@1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/backo2/-/backo2-1.0.2.tgz#31ab1ac8b129363463e35b3ebb69f4dfcfba7947" - integrity sha1-MasayLEpNjRj41s+u2n038+6eUc= - -balanced-match@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.0.tgz#89b4d199ab2bee49de164ea02b89ce462d71b767" - integrity sha1-ibTRmasr7kneFk6gK4nORi1xt2c= - -base64-arraybuffer@0.1.4: - version "0.1.4" - resolved "https://registry.yarnpkg.com/base64-arraybuffer/-/base64-arraybuffer-0.1.4.tgz#9818c79e059b1355f97e0428a017c838e90ba812" - integrity sha1-mBjHngWbE1X5fgQooBfIOOkLqBI= - -base64-js@^1.0.2: - version "1.5.1" - resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a" - integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA== - -base64id@2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/base64id/-/base64id-2.0.0.tgz#2770ac6bc47d312af97a8bf9a634342e0cd25cb6" - integrity sha512-lGe34o6EHj9y3Kts9R4ZYs/Gr+6N7MCaMlIFA3F1R2O5/m7K06AxfSeO5530PEERE6/WyEg3lsuyw4GHlPZHog== - -base@^0.11.1: - version "0.11.2" - resolved "https://registry.yarnpkg.com/base/-/base-0.11.2.tgz#7bde5ced145b6d551a90db87f83c558b4eb48a8f" - integrity sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg== - dependencies: - cache-base "^1.0.1" - class-utils "^0.3.5" - component-emitter "^1.2.1" - define-property "^1.0.0" - isobject "^3.0.1" - mixin-deep "^1.2.0" - pascalcase "^0.1.1" - -batch@0.6.1: - version "0.6.1" - resolved "https://registry.yarnpkg.com/batch/-/batch-0.6.1.tgz#dc34314f4e679318093fc760272525f94bf25c16" - integrity sha1-3DQxT05nkxgJP8dgJyUl+UvyXBY= - -bcrypt-pbkdf@^1.0.0: - version "1.0.2" - resolved "https://registry.yarnpkg.com/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz#a4301d389b6a43f9b67ff3ca11a3f6637e360e9e" - integrity sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4= - dependencies: - tweetnacl "^0.14.3" - -big.js@^5.2.2: - version "5.2.2" - resolved "https://registry.yarnpkg.com/big.js/-/big.js-5.2.2.tgz#65f0af382f578bcdc742bd9c281e9cb2d7768328" - integrity sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ== - -binary-extensions@^1.0.0: - version "1.13.1" - resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-1.13.1.tgz#598afe54755b2868a5330d2aff9d4ebb53209b65" - integrity sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw== - -binary-extensions@^2.0.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.2.0.tgz#75f502eeaf9ffde42fc98829645be4ea76bd9e2d" - integrity sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA== - -bindings@^1.5.0: - version "1.5.0" - resolved "https://registry.yarnpkg.com/bindings/-/bindings-1.5.0.tgz#10353c9e945334bc0511a6d90b38fbc7c9c504df" - integrity sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ== - dependencies: - file-uri-to-path "1.0.0" - -blob@0.0.5: - version "0.0.5" - resolved "https://registry.yarnpkg.com/blob/-/blob-0.0.5.tgz#d680eeef25f8cd91ad533f5b01eed48e64caf683" - integrity sha512-gaqbzQPqOoamawKg0LGVd7SzLgXS+JH61oWprSLH+P+abTczqJbhTR8CmJ2u9/bUYNmHTGJx/UEmn6doAvvuig== - -blocking-proxy@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/blocking-proxy/-/blocking-proxy-1.0.1.tgz#81d6fd1fe13a4c0d6957df7f91b75e98dac40cb2" - integrity sha512-KE8NFMZr3mN2E0HcvCgRtX7DjhiIQrwle+nSVJVC/yqFb9+xznHl2ZcoBp2L9qzkI4t4cBFJ1efXF8Dwi132RA== - dependencies: - minimist "^1.2.0" - -bluebird@^3.5.1, bluebird@^3.5.3, bluebird@^3.5.5: - version "3.7.2" - resolved "https://registry.yarnpkg.com/bluebird/-/bluebird-3.7.2.tgz#9f229c15be272454ffa973ace0dbee79a1b0c36f" - integrity sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg== - -bn.js@^4.0.0, bn.js@^4.1.0, bn.js@^4.11.9: - version "4.12.0" - resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-4.12.0.tgz#775b3f278efbb9718eec7361f483fb36fbbfea88" - integrity sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA== - -bn.js@^5.0.0, bn.js@^5.1.1: - version "5.2.0" - resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-5.2.0.tgz#358860674396c6997771a9d051fcc1b57d4ae002" - integrity sha512-D7iWRBvnZE8ecXiLj/9wbxH7Tk79fAh8IHaTNq1RWRixsS02W+5qS+iE9yq6RYl0asXx5tw0bLhmT5pIfbSquw== - -body-parser@1.19.0, body-parser@^1.19.0: - version "1.19.0" - resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.19.0.tgz#96b2709e57c9c4e09a6fd66a8fd979844f69f08a" - integrity sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw== - dependencies: - bytes "3.1.0" - content-type "~1.0.4" - debug "2.6.9" - depd "~1.1.2" - http-errors "1.7.2" - iconv-lite "0.4.24" - on-finished "~2.3.0" - qs "6.7.0" - raw-body "2.4.0" - type-is "~1.6.17" - -bonjour@^3.5.0: - version "3.5.0" - resolved "https://registry.yarnpkg.com/bonjour/-/bonjour-3.5.0.tgz#8e890a183d8ee9a2393b3844c691a42bcf7bc9f5" - integrity sha1-jokKGD2O6aI5OzhExpGkK897yfU= - dependencies: - array-flatten "^2.1.0" - deep-equal "^1.0.1" - dns-equal "^1.0.0" - dns-txt "^2.0.2" - multicast-dns "^6.0.1" - multicast-dns-service-types "^1.1.0" - -boolbase@^1.0.0, boolbase@~1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/boolbase/-/boolbase-1.0.0.tgz#68dff5fbe60c51eb37725ea9e3ed310dcc1e776e" - integrity sha1-aN/1++YMUes3cl6p4+0xDcwed24= - -boolean@^3.0.1: - version "3.0.2" - resolved "https://registry.yarnpkg.com/boolean/-/boolean-3.0.2.tgz#df1baa18b6a2b0e70840475e1d93ec8fe75b2570" - integrity sha512-RwywHlpCRc3/Wh81MiCKun4ydaIFyW5Ea6JbL6sRCVx5q5irDw7pMXBUFYF/jArQ6YrG36q0kpovc9P/Kd3I4g== - -brace-expansion@^1.1.7: - version "1.1.11" - resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" - integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA== - dependencies: - balanced-match "^1.0.0" - concat-map "0.0.1" - -braces@^2.3.1, braces@^2.3.2: - version "2.3.2" - resolved "https://registry.yarnpkg.com/braces/-/braces-2.3.2.tgz#5979fd3f14cd531565e5fa2df1abfff1dfaee729" - integrity sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w== - dependencies: - arr-flatten "^1.1.0" - array-unique "^0.3.2" - extend-shallow "^2.0.1" - fill-range "^4.0.0" - isobject "^3.0.1" - repeat-element "^1.1.2" - snapdragon "^0.8.1" - snapdragon-node "^2.0.1" - split-string "^3.0.2" - to-regex "^3.0.1" - -braces@^3.0.1, braces@^3.0.2, braces@~3.0.2: - version "3.0.2" - resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107" - integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A== - dependencies: - fill-range "^7.0.1" - -brorand@^1.0.1, brorand@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/brorand/-/brorand-1.1.0.tgz#12c25efe40a45e3c323eb8675a0a0ce57b22371f" - integrity sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8= - -browserify-aes@^1.0.0, browserify-aes@^1.0.4: - version "1.2.0" - resolved "https://registry.yarnpkg.com/browserify-aes/-/browserify-aes-1.2.0.tgz#326734642f403dabc3003209853bb70ad428ef48" - integrity sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA== - dependencies: - buffer-xor "^1.0.3" - cipher-base "^1.0.0" - create-hash "^1.1.0" - evp_bytestokey "^1.0.3" - inherits "^2.0.1" - safe-buffer "^5.0.1" - -browserify-cipher@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/browserify-cipher/-/browserify-cipher-1.0.1.tgz#8d6474c1b870bfdabcd3bcfcc1934a10e94f15f0" - integrity sha512-sPhkz0ARKbf4rRQt2hTpAHqn47X3llLkUGn+xEJzLjwY8LRs2p0v7ljvI5EyoRO/mexrNunNECisZs+gw2zz1w== - dependencies: - browserify-aes "^1.0.4" - browserify-des "^1.0.0" - evp_bytestokey "^1.0.0" - -browserify-des@^1.0.0: - version "1.0.2" - resolved "https://registry.yarnpkg.com/browserify-des/-/browserify-des-1.0.2.tgz#3af4f1f59839403572f1c66204375f7a7f703e9c" - integrity sha512-BioO1xf3hFwz4kc6iBhI3ieDFompMhrMlnDFC4/0/vd5MokpuAc3R+LYbwTA9A5Yc9pq9UYPqffKpW2ObuwX5A== - dependencies: - cipher-base "^1.0.1" - des.js "^1.0.0" - inherits "^2.0.1" - safe-buffer "^5.1.2" - -browserify-rsa@^4.0.0, browserify-rsa@^4.0.1: - version "4.1.0" - resolved "https://registry.yarnpkg.com/browserify-rsa/-/browserify-rsa-4.1.0.tgz#b2fd06b5b75ae297f7ce2dc651f918f5be158c8d" - integrity sha512-AdEER0Hkspgno2aR97SAf6vi0y0k8NuOpGnVH3O99rcA5Q6sh8QxcngtHuJ6uXwnfAXNM4Gn1Gb7/MV1+Ymbog== - dependencies: - bn.js "^5.0.0" - randombytes "^2.0.1" - -browserify-sign@^4.0.0: - version "4.2.1" - resolved "https://registry.yarnpkg.com/browserify-sign/-/browserify-sign-4.2.1.tgz#eaf4add46dd54be3bb3b36c0cf15abbeba7956c3" - integrity sha512-/vrA5fguVAKKAVTNJjgSm1tRQDHUU6DbwO9IROu/0WAzC8PKhucDSh18J0RMvVeHAn5puMd+QHC2erPRNf8lmg== - dependencies: - bn.js "^5.1.1" - browserify-rsa "^4.0.1" - create-hash "^1.2.0" - create-hmac "^1.1.7" - elliptic "^6.5.3" - inherits "^2.0.4" - parse-asn1 "^5.1.5" - readable-stream "^3.6.0" - safe-buffer "^5.2.0" - -browserify-zlib@^0.2.0: - version "0.2.0" - resolved "https://registry.yarnpkg.com/browserify-zlib/-/browserify-zlib-0.2.0.tgz#2869459d9aa3be245fe8fe2ca1f46e2e7f54d73f" - integrity sha512-Z942RysHXmJrhqk88FmKBVq/v5tqmSkDz7p54G/MGyjMnCFFnC79XWNbg+Vta8W6Wb2qtSZTSxIGkJrRpCFEiA== - dependencies: - pako "~1.0.5" - -browserslist@^4.0.0, browserslist@^4.12.0, browserslist@^4.14.5, browserslist@^4.16.3, browserslist@^4.9.1: - version "4.16.3" - resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.16.3.tgz#340aa46940d7db878748567c5dea24a48ddf3717" - integrity sha512-vIyhWmIkULaq04Gt93txdh+j02yX/JzlyhLYbV3YQCn/zvES3JnY7TifHHvvr1w5hTDluNKMkV05cs4vy8Q7sw== - dependencies: - caniuse-lite "^1.0.30001181" - colorette "^1.2.1" - electron-to-chromium "^1.3.649" - escalade "^3.1.1" - node-releases "^1.1.70" - -browserstack@^1.5.1: - version "1.6.1" - resolved "https://registry.yarnpkg.com/browserstack/-/browserstack-1.6.1.tgz#e051f9733ec3b507659f395c7a4765a1b1e358b3" - integrity sha512-GxtFjpIaKdbAyzHfFDKixKO8IBT7wR3NjbzrGc78nNs/Ciys9wU3/nBtsqsWv5nDSrdI5tz0peKuzCPuNXNUiw== - dependencies: - https-proxy-agent "^2.2.1" - -buffer-crc32@~0.2.3: - version "0.2.13" - resolved "https://registry.yarnpkg.com/buffer-crc32/-/buffer-crc32-0.2.13.tgz#0d333e3f00eac50aa1454abd30ef8c2a5d9a7242" - integrity sha1-DTM+PwDqxQqhRUq9MO+MKl2ackI= - -buffer-from@^1.0.0: - version "1.1.1" - resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.1.tgz#32713bc028f75c02fdb710d7c7bcec1f2c6070ef" - integrity sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A== - -buffer-indexof@^1.0.0: - version "1.1.1" - resolved "https://registry.yarnpkg.com/buffer-indexof/-/buffer-indexof-1.1.1.tgz#52fabcc6a606d1a00302802648ef68f639da268c" - integrity sha512-4/rOEg86jivtPTeOUUT61jJO1Ya1TrR/OkqCSZDyq84WJh3LuuiphBYJN+fm5xufIk4XAFcEwte/8WzC8If/1g== - -buffer-xor@^1.0.3: - version "1.0.3" - resolved "https://registry.yarnpkg.com/buffer-xor/-/buffer-xor-1.0.3.tgz#26e61ed1422fb70dd42e6e36729ed51d855fe8d9" - integrity sha1-JuYe0UIvtw3ULm42cp7VHYVf6Nk= - -buffer@^4.3.0: - version "4.9.2" - resolved "https://registry.yarnpkg.com/buffer/-/buffer-4.9.2.tgz#230ead344002988644841ab0244af8c44bbe3ef8" - integrity sha512-xq+q3SRMOxGivLhBNaUdC64hDTQwejJ+H0T/NB1XMtTVEwNTrfFF3gAxiyW0Bu/xWEGhjVKgUcMhCrUy2+uCWg== - dependencies: - base64-js "^1.0.2" - ieee754 "^1.1.4" - isarray "^1.0.0" - -builtin-modules@^1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/builtin-modules/-/builtin-modules-1.1.1.tgz#270f076c5a72c02f5b65a47df94c5fe3a278892f" - integrity sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8= - -builtin-status-codes@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/builtin-status-codes/-/builtin-status-codes-3.0.0.tgz#85982878e21b98e1c66425e03d0174788f569ee8" - integrity sha1-hZgoeOIbmOHGZCXgPQF0eI9Wnug= - -builtins@^1.0.3: - version "1.0.3" - resolved "https://registry.yarnpkg.com/builtins/-/builtins-1.0.3.tgz#cb94faeb61c8696451db36534e1422f94f0aee88" - integrity sha1-y5T662HIaWRR2zZTThQi+U8K7og= - -bytes@3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.0.0.tgz#d32815404d689699f85a4ea4fa8755dd13a96048" - integrity sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg= - -bytes@3.1.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.0.tgz#f6cf7933a360e0588fa9fde85651cdc7f805d1f6" - integrity sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg== - -cacache@15.0.5, cacache@^15.0.5: - version "15.0.5" - resolved "https://registry.yarnpkg.com/cacache/-/cacache-15.0.5.tgz#69162833da29170d6732334643c60e005f5f17d0" - integrity sha512-lloiL22n7sOjEEXdL8NAjTgv9a1u43xICE9/203qonkZUCj5X1UEWIdf2/Y0d6QcCtMzbKQyhrcDbdvlZTs/+A== - dependencies: - "@npmcli/move-file" "^1.0.1" - chownr "^2.0.0" - fs-minipass "^2.0.0" - glob "^7.1.4" - infer-owner "^1.0.4" - lru-cache "^6.0.0" - minipass "^3.1.1" - minipass-collect "^1.0.2" - minipass-flush "^1.0.5" - minipass-pipeline "^1.2.2" - mkdirp "^1.0.3" - p-map "^4.0.0" - promise-inflight "^1.0.1" - rimraf "^3.0.2" - ssri "^8.0.0" - tar "^6.0.2" - unique-filename "^1.1.1" - -cacache@^12.0.0, cacache@^12.0.2: - version "12.0.4" - resolved "https://registry.yarnpkg.com/cacache/-/cacache-12.0.4.tgz#668bcbd105aeb5f1d92fe25570ec9525c8faa40c" - integrity sha512-a0tMB40oefvuInr4Cwb3GerbL9xTj1D5yg0T5xrjGCGyfvbxseIXX7BAO/u/hIXdafzOI5JC3wDwHyf24buOAQ== - dependencies: - bluebird "^3.5.5" - chownr "^1.1.1" - figgy-pudding "^3.5.1" - glob "^7.1.4" - graceful-fs "^4.1.15" - infer-owner "^1.0.3" - lru-cache "^5.1.1" - mississippi "^3.0.0" - mkdirp "^0.5.1" - move-concurrently "^1.0.1" - promise-inflight "^1.0.1" - rimraf "^2.6.3" - ssri "^6.0.1" - unique-filename "^1.1.1" - y18n "^4.0.0" - -cache-base@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/cache-base/-/cache-base-1.0.1.tgz#0a7f46416831c8b662ee36fe4e7c59d76f666ab2" - integrity sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ== - dependencies: - collection-visit "^1.0.0" - component-emitter "^1.2.1" - get-value "^2.0.6" - has-value "^1.0.0" - isobject "^3.0.1" - set-value "^2.0.0" - to-object-path "^0.3.0" - union-value "^1.0.0" - unset-value "^1.0.0" - -cacheable-request@^6.0.0: - version "6.1.0" - resolved "https://registry.yarnpkg.com/cacheable-request/-/cacheable-request-6.1.0.tgz#20ffb8bd162ba4be11e9567d823db651052ca912" - integrity sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg== - dependencies: - clone-response "^1.0.2" - get-stream "^5.1.0" - http-cache-semantics "^4.0.0" - keyv "^3.0.0" - lowercase-keys "^2.0.0" - normalize-url "^4.1.0" - responselike "^1.0.2" - -call-bind@^1.0.0, call-bind@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.2.tgz#b1d4e89e688119c3c9a903ad30abb2f6a919be3c" - integrity sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA== - dependencies: - function-bind "^1.1.1" - get-intrinsic "^1.0.2" - -caller-callsite@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/caller-callsite/-/caller-callsite-2.0.0.tgz#847e0fce0a223750a9a027c54b33731ad3154134" - integrity sha1-hH4PzgoiN1CpoCfFSzNzGtMVQTQ= - dependencies: - callsites "^2.0.0" - -caller-path@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/caller-path/-/caller-path-2.0.0.tgz#468f83044e369ab2010fac5f06ceee15bb2cb1f4" - integrity sha1-Ro+DBE42mrIBD6xfBs7uFbsssfQ= - dependencies: - caller-callsite "^2.0.0" - -callsites@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/callsites/-/callsites-2.0.0.tgz#06eb84f00eea413da86affefacbffb36093b3c50" - integrity sha1-BuuE8A7qQT2oav/vrL/7Ngk7PFA= - -callsites@^3.0.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73" - integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ== - -camelcase@5.3.1, camelcase@^5.0.0: - version "5.3.1" - resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-5.3.1.tgz#e3c9b31569e106811df242f715725a1f4c494320" - integrity sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg== - -camelcase@^6.0.0: - version "6.2.0" - resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-6.2.0.tgz#924af881c9d525ac9d87f40d964e5cea982a1809" - integrity sha512-c7wVvbw3f37nuobQNtgsgG9POC9qMbNuMQmTCqZv23b6MIz0fcYpBiOlv9gEN/hdLdnZTDQhg6e9Dq5M1vKvfg== - -caniuse-api@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/caniuse-api/-/caniuse-api-3.0.0.tgz#5e4d90e2274961d46291997df599e3ed008ee4c0" - integrity sha512-bsTwuIg/BZZK/vreVTYYbSWoe2F+71P7K5QGEX+pT250DZbfU1MQ5prOKpPR+LL6uWKK3KMwMCAS74QB3Um1uw== - dependencies: - browserslist "^4.0.0" - caniuse-lite "^1.0.0" - lodash.memoize "^4.1.2" - lodash.uniq "^4.5.0" - -caniuse-lite@^1.0.0, caniuse-lite@^1.0.30001032, caniuse-lite@^1.0.30001109, caniuse-lite@^1.0.30001181: - version "1.0.30001199" - resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001199.tgz#062afccaad21023e2e647d767bac4274b8b8fd7f" - integrity sha512-ifbK2eChUCFUwGhlEzIoVwzFt1+iriSjyKKFYNfv6hN34483wyWpLLavYQXhnR036LhkdUYaSDpHg1El++VgHQ== - -canonical-path@1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/canonical-path/-/canonical-path-1.0.0.tgz#fcb470c23958def85081856be7a86e904f180d1d" - integrity sha512-feylzsbDxi1gPZ1IjystzIQZagYYLvfKrSuygUCgf7z6x790VEzze5QEkdSV1U58RA7Hi0+v6fv4K54atOzATg== - -caseless@~0.12.0: - version "0.12.0" - resolved "https://registry.yarnpkg.com/caseless/-/caseless-0.12.0.tgz#1b681c21ff84033c826543090689420d187151dc" - integrity sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw= - -chalk@^1.1.1, chalk@^1.1.3: - version "1.1.3" - resolved "https://registry.yarnpkg.com/chalk/-/chalk-1.1.3.tgz#a8115c55e4a702fe4d150abd3872822a7e09fc98" - integrity sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg= - dependencies: - ansi-styles "^2.2.1" - escape-string-regexp "^1.0.2" - has-ansi "^2.0.0" - strip-ansi "^3.0.0" - supports-color "^2.0.0" - -chalk@^2.0.0, chalk@^2.0.1, chalk@^2.3.0, chalk@^2.4.1, chalk@^2.4.2: - version "2.4.2" - resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz#cd42541677a54333cf541a49108c1432b44c9424" - integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ== - dependencies: - ansi-styles "^3.2.1" - escape-string-regexp "^1.0.5" - supports-color "^5.3.0" - -chalk@^4.0.0, chalk@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.0.tgz#4e14870a618d9e2edd97dd8345fd9d9dc315646a" - integrity sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A== - dependencies: - ansi-styles "^4.1.0" - supports-color "^7.1.0" - -chardet@^0.7.0: - version "0.7.0" - resolved "https://registry.yarnpkg.com/chardet/-/chardet-0.7.0.tgz#90094849f0937f2eedc2425d0d28a9e5f0cbad9e" - integrity sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA== - -"chokidar@>=2.0.0 <4.0.0", chokidar@^3.0.0, chokidar@^3.4.1: - version "3.5.1" - resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-3.5.1.tgz#ee9ce7bbebd2b79f49f304799d5468e31e14e68a" - integrity sha512-9+s+Od+W0VJJzawDma/gvBNQqkTiqYTWLuZoyAsivsI4AaWTCzHG06/TMjsf1cYe9Cb97UCEhjz7HvnPk2p/tw== - dependencies: - anymatch "~3.1.1" - braces "~3.0.2" - glob-parent "~5.1.0" - is-binary-path "~2.1.0" - is-glob "~4.0.1" - normalize-path "~3.0.0" - readdirp "~3.5.0" - optionalDependencies: - fsevents "~2.3.1" - -chokidar@^2.1.8: - version "2.1.8" - resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-2.1.8.tgz#804b3a7b6a99358c3c5c61e71d8728f041cff917" - integrity sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg== - dependencies: - anymatch "^2.0.0" - async-each "^1.0.1" - braces "^2.3.2" - glob-parent "^3.1.0" - inherits "^2.0.3" - is-binary-path "^1.0.0" - is-glob "^4.0.0" - normalize-path "^3.0.0" - path-is-absolute "^1.0.0" - readdirp "^2.2.1" - upath "^1.1.1" - optionalDependencies: - fsevents "^1.2.7" - -chownr@^1.1.1, chownr@^1.1.2: - version "1.1.4" - resolved "https://registry.yarnpkg.com/chownr/-/chownr-1.1.4.tgz#6fc9d7b42d32a583596337666e7d08084da2cc6b" - integrity sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg== - -chownr@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/chownr/-/chownr-2.0.0.tgz#15bfbe53d2eab4cf70f18a8cd68ebe5b3cb1dece" - integrity sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ== - -chrome-trace-event@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/chrome-trace-event/-/chrome-trace-event-1.0.2.tgz#234090ee97c7d4ad1a2c4beae27505deffc608a4" - integrity sha512-9e/zx1jw7B4CO+c/RXoCsfg/x1AfUBioy4owYH0bJprEYAx5hRFLRhWBqHAG57D0ZM4H7vxbP7bPe0VwhQRYDQ== - dependencies: - tslib "^1.9.0" - -cipher-base@^1.0.0, cipher-base@^1.0.1, cipher-base@^1.0.3: - version "1.0.4" - resolved "https://registry.yarnpkg.com/cipher-base/-/cipher-base-1.0.4.tgz#8760e4ecc272f4c363532f926d874aae2c1397de" - integrity sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q== - dependencies: - inherits "^2.0.1" - safe-buffer "^5.0.1" - -circular-dependency-plugin@5.2.0: - version "5.2.0" - resolved "https://registry.yarnpkg.com/circular-dependency-plugin/-/circular-dependency-plugin-5.2.0.tgz#e09dbc2dd3e2928442403e2d45b41cea06bc0a93" - integrity sha512-7p4Kn/gffhQaavNfyDFg7LS5S/UT1JAjyGd4UqR2+jzoYF02eDkj0Ec3+48TsIa4zghjLY87nQHIh/ecK9qLdw== - -class-utils@^0.3.5: - version "0.3.6" - resolved "https://registry.yarnpkg.com/class-utils/-/class-utils-0.3.6.tgz#f93369ae8b9a7ce02fd41faad0ca83033190c463" - integrity sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg== - dependencies: - arr-union "^3.1.0" - define-property "^0.2.5" - isobject "^3.0.0" - static-extend "^0.1.1" - -clean-stack@^2.0.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/clean-stack/-/clean-stack-2.2.0.tgz#ee8472dbb129e727b31e8a10a427dee9dfe4008b" - integrity sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A== - -cli-cursor@^3.1.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-3.1.0.tgz#264305a7ae490d1d03bf0c9ba7c925d1753af307" - integrity sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw== - dependencies: - restore-cursor "^3.1.0" - -cli-spinners@^2.4.0: - version "2.5.0" - resolved "https://registry.yarnpkg.com/cli-spinners/-/cli-spinners-2.5.0.tgz#12763e47251bf951cb75c201dfa58ff1bcb2d047" - integrity sha512-PC+AmIuK04E6aeSs/pUccSujsTzBhu4HzC2dL+CfJB/Jcc2qTRbEwZQDfIUpt2Xl8BodYBEq8w4fc0kU2I9DjQ== - -cli-width@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/cli-width/-/cli-width-3.0.0.tgz#a2f48437a2caa9a22436e794bf071ec9e61cedf6" - integrity sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw== - -cliui@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/cliui/-/cliui-5.0.0.tgz#deefcfdb2e800784aa34f46fa08e06851c7bbbc5" - integrity sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA== - dependencies: - string-width "^3.1.0" - strip-ansi "^5.2.0" - wrap-ansi "^5.1.0" - -cliui@^6.0.0: - version "6.0.0" - resolved "https://registry.yarnpkg.com/cliui/-/cliui-6.0.0.tgz#511d702c0c4e41ca156d7d0e96021f23e13225b1" - integrity sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ== - dependencies: - string-width "^4.2.0" - strip-ansi "^6.0.0" - wrap-ansi "^6.2.0" - -cliui@^7.0.2: - version "7.0.4" - resolved "https://registry.yarnpkg.com/cliui/-/cliui-7.0.4.tgz#a0265ee655476fc807aea9df3df8df7783808b4f" - integrity sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ== - dependencies: - string-width "^4.2.0" - strip-ansi "^6.0.0" - wrap-ansi "^7.0.0" - -clone-deep@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/clone-deep/-/clone-deep-4.0.1.tgz#c19fd9bdbbf85942b4fd979c84dcf7d5f07c2387" - integrity sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ== - dependencies: - is-plain-object "^2.0.4" - kind-of "^6.0.2" - shallow-clone "^3.0.0" - -clone-response@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/clone-response/-/clone-response-1.0.2.tgz#d1dc973920314df67fbeb94223b4ee350239e96b" - integrity sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws= - dependencies: - mimic-response "^1.0.0" - -clone@^1.0.2: - version "1.0.4" - resolved "https://registry.yarnpkg.com/clone/-/clone-1.0.4.tgz#da309cc263df15994c688ca902179ca3c7cd7c7e" - integrity sha1-2jCcwmPfFZlMaIypAheco8fNfH4= - -coa@^2.0.2: - version "2.0.2" - resolved "https://registry.yarnpkg.com/coa/-/coa-2.0.2.tgz#43f6c21151b4ef2bf57187db0d73de229e3e7ec3" - integrity sha512-q5/jG+YQnSy4nRTV4F7lPepBJZ8qBNJJDBuJdoejDyLXgmL7IEo+Le2JDZudFTFt7mrCqIRaSjws4ygRCTCAXA== - dependencies: - "@types/q" "^1.5.1" - chalk "^2.4.1" - q "^1.1.2" - -codelyzer@6.0.0: - version "6.0.0" - resolved "https://registry.yarnpkg.com/codelyzer/-/codelyzer-6.0.0.tgz#50c98581cc2890e0e9a9f93878dc317115d836ed" - integrity sha512-edJIQCIcxD9DhVSyBEdJ38AbLikm515Wl91t5RDGNT88uA6uQdTm4phTWfn9JhzAI8kXNUcfYyAE90lJElpGtA== - dependencies: - "@angular/compiler" "9.0.0" - "@angular/core" "9.0.0" - app-root-path "^3.0.0" - aria-query "^3.0.0" - axobject-query "2.0.2" - css-selector-tokenizer "^0.7.1" - cssauron "^1.4.0" - damerau-levenshtein "^1.0.4" - rxjs "^6.5.3" - semver-dsl "^1.0.1" - source-map "^0.5.7" - sprintf-js "^1.1.2" - tslib "^1.10.0" - zone.js "~0.10.3" - -collection-visit@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/collection-visit/-/collection-visit-1.0.0.tgz#4bc0373c164bc3291b4d368c829cf1a80a59dca0" - integrity sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA= - dependencies: - map-visit "^1.0.0" - object-visit "^1.0.0" - -color-convert@^1.9.0, color-convert@^1.9.1: - version "1.9.3" - resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.3.tgz#bb71850690e1f136567de629d2d5471deda4c1e8" - integrity sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg== - dependencies: - color-name "1.1.3" - -color-convert@^2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3" - integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ== - dependencies: - color-name "~1.1.4" - -color-name@1.1.3: - version "1.1.3" - resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25" - integrity sha1-p9BVi9icQveV3UIyj3QIMcpTvCU= - -color-name@^1.0.0, color-name@~1.1.4: - version "1.1.4" - resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" - integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== - -color-string@^1.5.4: - version "1.5.5" - resolved "https://registry.yarnpkg.com/color-string/-/color-string-1.5.5.tgz#65474a8f0e7439625f3d27a6a19d89fc45223014" - integrity sha512-jgIoum0OfQfq9Whcfc2z/VhCNcmQjWbey6qBX0vqt7YICflUmBCh9E9CiQD5GSJ+Uehixm3NUwHVhqUAWRivZg== - dependencies: - color-name "^1.0.0" - simple-swizzle "^0.2.2" - -color@^3.0.0: - version "3.1.3" - resolved "https://registry.yarnpkg.com/color/-/color-3.1.3.tgz#ca67fb4e7b97d611dcde39eceed422067d91596e" - integrity sha512-xgXAcTHa2HeFCGLE9Xs/R82hujGtu9Jd9x4NW3T34+OMs7VoPsjwzRczKHvTAHeJwWFwX5j15+MgAppE8ztObQ== - dependencies: - color-convert "^1.9.1" - color-string "^1.5.4" - -colorette@^1.2.1: - version "1.2.2" - resolved "https://registry.yarnpkg.com/colorette/-/colorette-1.2.2.tgz#cbcc79d5e99caea2dbf10eb3a26fd8b3e6acfa94" - integrity sha512-MKGMzyfeuutC/ZJ1cba9NqcNpfeqMUcYmyF1ZFY6/Cn7CNSAKx6a+s48sqLqyAiZuaP2TcqMhoo+dlwFnVxT9w== - -colors@1.4.0, colors@^1.4.0: - version "1.4.0" - resolved "https://registry.yarnpkg.com/colors/-/colors-1.4.0.tgz#c50491479d4c1bdaed2c9ced32cf7c7dc2360f78" - integrity sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA== - -combined-stream@^1.0.6, combined-stream@~1.0.6: - version "1.0.8" - resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f" - integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg== - dependencies: - delayed-stream "~1.0.0" - -commander@^2.11.0, commander@^2.12.1, commander@^2.20.0: - version "2.20.3" - resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.3.tgz#fd485e84c03eb4881c20722ba48035e8531aeb33" - integrity sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ== - -commondir@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/commondir/-/commondir-1.0.1.tgz#ddd800da0c66127393cca5950ea968a3aaf1253b" - integrity sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs= - -component-bind@1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/component-bind/-/component-bind-1.0.0.tgz#00c608ab7dcd93897c0009651b1d3a8e1e73bbd1" - integrity sha1-AMYIq33Nk4l8AAllGx06jh5zu9E= - -component-emitter@1.2.1: - version "1.2.1" - resolved "https://registry.yarnpkg.com/component-emitter/-/component-emitter-1.2.1.tgz#137918d6d78283f7df7a6b7c5a63e140e69425e6" - integrity sha1-E3kY1teCg/ffemt8WmPhQOaUJeY= - -component-emitter@^1.2.1, component-emitter@~1.3.0: - version "1.3.0" - resolved "https://registry.yarnpkg.com/component-emitter/-/component-emitter-1.3.0.tgz#16e4070fba8ae29b679f2215853ee181ab2eabc0" - integrity sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg== - -component-inherit@0.0.3: - version "0.0.3" - resolved "https://registry.yarnpkg.com/component-inherit/-/component-inherit-0.0.3.tgz#645fc4adf58b72b649d5cae65135619db26ff143" - integrity sha1-ZF/ErfWLcrZJ1crmUTVhnbJv8UM= - -compose-function@3.0.3: - version "3.0.3" - resolved "https://registry.yarnpkg.com/compose-function/-/compose-function-3.0.3.tgz#9ed675f13cc54501d30950a486ff6a7ba3ab185f" - integrity sha1-ntZ18TzFRQHTCVCkhv9qe6OrGF8= - dependencies: - arity-n "^1.0.4" - -compressible@~2.0.16: - version "2.0.18" - resolved "https://registry.yarnpkg.com/compressible/-/compressible-2.0.18.tgz#af53cca6b070d4c3c0750fbd77286a6d7cc46fba" - integrity sha512-AF3r7P5dWxL8MxyITRMlORQNaOA2IkAFaTr4k7BUumjPtRpGDTZpl0Pb1XCO6JeDCBdp126Cgs9sMxqSjgYyRg== - dependencies: - mime-db ">= 1.43.0 < 2" - -compression@^1.7.4: - version "1.7.4" - resolved "https://registry.yarnpkg.com/compression/-/compression-1.7.4.tgz#95523eff170ca57c29a0ca41e6fe131f41e5bb8f" - integrity sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ== - dependencies: - accepts "~1.3.5" - bytes "3.0.0" - compressible "~2.0.16" - debug "2.6.9" - on-headers "~1.0.2" - safe-buffer "5.1.2" - vary "~1.1.2" - -concat-map@0.0.1: - version "0.0.1" - resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" - integrity sha1-2Klr13/Wjfd5OnMDajug1UBdR3s= - -concat-stream@^1.5.0, concat-stream@^1.6.2: - version "1.6.2" - resolved "https://registry.yarnpkg.com/concat-stream/-/concat-stream-1.6.2.tgz#904bdf194cd3122fc675c77fc4ac3d4ff0fd1a34" - integrity sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw== - dependencies: - buffer-from "^1.0.0" - inherits "^2.0.3" - readable-stream "^2.2.2" - typedarray "^0.0.6" - -config-chain@^1.1.11: - version "1.1.12" - resolved "https://registry.yarnpkg.com/config-chain/-/config-chain-1.1.12.tgz#0fde8d091200eb5e808caf25fe618c02f48e4efa" - integrity sha512-a1eOIcu8+7lUInge4Rpf/n4Krkf3Dd9lqhljRzII1/Zno/kRtUWnznPO3jOKBmTEktkt3fkxisUcivoj0ebzoA== - dependencies: - ini "^1.3.4" - proto-list "~1.2.1" - -connect-history-api-fallback@^1.6.0: - version "1.6.0" - resolved "https://registry.yarnpkg.com/connect-history-api-fallback/-/connect-history-api-fallback-1.6.0.tgz#8b32089359308d111115d81cad3fceab888f97bc" - integrity sha512-e54B99q/OUoH64zYYRf3HBP5z24G38h5D3qXu23JGRoigpX5Ss4r9ZnDk3g0Z8uQC2x2lPaJ+UlWBc1ZWBWdLg== - -connect@^3.7.0: - version "3.7.0" - resolved "https://registry.yarnpkg.com/connect/-/connect-3.7.0.tgz#5d49348910caa5e07a01800b030d0c35f20484f8" - integrity sha512-ZqRXc+tZukToSNmh5C2iWMSoV3X1YUcPbqEM4DkEG5tNQXrQUZCNVGGv3IuicnkMtPfGf3Xtp8WCXs295iQ1pQ== - dependencies: - debug "2.6.9" - finalhandler "1.1.2" - parseurl "~1.3.3" - utils-merge "1.0.1" - -console-browserify@^1.1.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/console-browserify/-/console-browserify-1.2.0.tgz#67063cef57ceb6cf4993a2ab3a55840ae8c49336" - integrity sha512-ZMkYO/LkF17QvCPqM0gxw8yUzigAOZOSWSHg91FH6orS7vcEj5dVZTidN2fQ14yBSdg97RqhSNwLUXInd52OTA== - -constants-browserify@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/constants-browserify/-/constants-browserify-1.0.0.tgz#c20b96d8c617748aaf1c16021760cd27fcb8cb75" - integrity sha1-wguW2MYXdIqvHBYCF2DNJ/y4y3U= - -content-disposition@0.5.3: - version "0.5.3" - resolved "https://registry.yarnpkg.com/content-disposition/-/content-disposition-0.5.3.tgz#e130caf7e7279087c5616c2007d0485698984fbd" - integrity sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g== - dependencies: - safe-buffer "5.1.2" - -content-type@~1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/content-type/-/content-type-1.0.4.tgz#e138cc75e040c727b1966fe5e5f8c9aee256fe3b" - integrity sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA== - -convert-source-map@1.7.0, convert-source-map@^1.5.1, convert-source-map@^1.7.0: - version "1.7.0" - resolved "https://registry.yarnpkg.com/convert-source-map/-/convert-source-map-1.7.0.tgz#17a2cb882d7f77d3490585e2ce6c524424a3a442" - integrity sha512-4FJkXzKXEDB1snCFZlLP4gpC3JILicCpGbzG9f9G7tGqGCzETQ2hWPrcinA9oU4wtf2biUaEH5065UnMeR33oA== - dependencies: - safe-buffer "~5.1.1" - -convert-source-map@^0.3.3: - version "0.3.5" - resolved "https://registry.yarnpkg.com/convert-source-map/-/convert-source-map-0.3.5.tgz#f1d802950af7dd2631a1febe0596550c86ab3190" - integrity sha1-8dgClQr33SYxof6+BZZVDIarMZA= - -cookie-signature@1.0.6: - version "1.0.6" - resolved "https://registry.yarnpkg.com/cookie-signature/-/cookie-signature-1.0.6.tgz#e303a882b342cc3ee8ca513a79999734dab3ae2c" - integrity sha1-4wOogrNCzD7oylE6eZmXNNqzriw= - -cookie@0.4.0: - version "0.4.0" - resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.4.0.tgz#beb437e7022b3b6d49019d088665303ebe9c14ba" - integrity sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg== - -cookie@~0.4.1: - version "0.4.1" - resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.4.1.tgz#afd713fe26ebd21ba95ceb61f9a8116e50a537d1" - integrity sha512-ZwrFkGJxUR3EIoXtO+yVE69Eb7KlixbaeAWfBQB9vVsNn/o+Yw69gBWSSDK825hQNdN+wF8zELf3dFNl/kxkUA== - -copy-concurrently@^1.0.0: - version "1.0.5" - resolved "https://registry.yarnpkg.com/copy-concurrently/-/copy-concurrently-1.0.5.tgz#92297398cae34937fcafd6ec8139c18051f0b5e0" - integrity sha512-f2domd9fsVDFtaFcbaRZuYXwtdmnzqbADSwhSWYxYB/Q8zsdUUFMXVRwXGDMWmbEzAn1kdRrtI1T/KTFOL4X2A== - dependencies: - aproba "^1.1.1" - fs-write-stream-atomic "^1.0.8" - iferr "^0.1.5" - mkdirp "^0.5.1" - rimraf "^2.5.4" - run-queue "^1.0.0" - -copy-descriptor@^0.1.0: - version "0.1.1" - resolved "https://registry.yarnpkg.com/copy-descriptor/-/copy-descriptor-0.1.1.tgz#676f6eb3c39997c2ee1ac3a924fd6124748f578d" - integrity sha1-Z29us8OZl8LuGsOpJP1hJHSPV40= - -copy-webpack-plugin@6.2.1: - version "6.2.1" - resolved "https://registry.yarnpkg.com/copy-webpack-plugin/-/copy-webpack-plugin-6.2.1.tgz#8015e4d5c5e637ab7b39c76daa9e03c7a4bf1ae5" - integrity sha512-VH2ZTMIBsx4p++Lmpg77adZ0KUyM5gFR/9cuTrbneNnJlcQXUFvsNariPqq2dq2kV3F2skHiDGPQCyKWy1+U0Q== - dependencies: - cacache "^15.0.5" - fast-glob "^3.2.4" - find-cache-dir "^3.3.1" - glob-parent "^5.1.1" - globby "^11.0.1" - loader-utils "^2.0.0" - normalize-path "^3.0.0" - p-limit "^3.0.2" - schema-utils "^3.0.0" - serialize-javascript "^5.0.1" - webpack-sources "^1.4.3" - -core-js-compat@^3.6.2: - version "3.9.1" - resolved "https://registry.yarnpkg.com/core-js-compat/-/core-js-compat-3.9.1.tgz#4e572acfe90aff69d76d8c37759d21a5c59bb455" - integrity sha512-jXAirMQxrkbiiLsCx9bQPJFA6llDadKMpYrBJQJ3/c4/vsPP/fAf29h24tviRlvwUL6AmY5CHLu2GvjuYviQqA== - dependencies: - browserslist "^4.16.3" - semver "7.0.0" - -core-js@3.6.5: - version "3.6.5" - resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.6.5.tgz#7395dc273af37fb2e50e9bd3d9fe841285231d1a" - integrity sha512-vZVEEwZoIsI+vPEuoF9Iqf5H7/M3eeQqWlQnYa8FSKKePuYTf5MWnxb5SDAzCa60b3JBRS5g9b+Dq7b1y/RCrA== - -core-js@^3.6.5: - version "3.9.1" - resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.9.1.tgz#cec8de593db8eb2a85ffb0dbdeb312cb6e5460ae" - integrity sha512-gSjRvzkxQc1zjM/5paAmL4idJBFzuJoo+jDjF1tStYFMV2ERfD02HhahhCGXUyHxQRG4yFKVSdO6g62eoRMcDg== - -core-util-is@1.0.2, core-util-is@~1.0.0: - version "1.0.2" - resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7" - integrity sha1-tf1UIgqivFq1eqtxQMlAdUUDwac= - -cosmiconfig@^5.0.0: - version "5.2.1" - resolved "https://registry.yarnpkg.com/cosmiconfig/-/cosmiconfig-5.2.1.tgz#040f726809c591e77a17c0a3626ca45b4f168b1a" - integrity sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA== - dependencies: - import-fresh "^2.0.0" - is-directory "^0.3.1" - js-yaml "^3.13.1" - parse-json "^4.0.0" - -cosmiconfig@^7.0.0: - version "7.0.0" - resolved "https://registry.yarnpkg.com/cosmiconfig/-/cosmiconfig-7.0.0.tgz#ef9b44d773959cae63ddecd122de23853b60f8d3" - integrity sha512-pondGvTuVYDk++upghXJabWzL6Kxu6f26ljFw64Swq9v6sQPUL3EUlVDV56diOjpCayKihL6hVe8exIACU4XcA== - dependencies: - "@types/parse-json" "^4.0.0" - import-fresh "^3.2.1" - parse-json "^5.0.0" - path-type "^4.0.0" - yaml "^1.10.0" - -create-ecdh@^4.0.0: - version "4.0.4" - resolved "https://registry.yarnpkg.com/create-ecdh/-/create-ecdh-4.0.4.tgz#d6e7f4bffa66736085a0762fd3a632684dabcc4e" - integrity sha512-mf+TCx8wWc9VpuxfP2ht0iSISLZnt0JgWlrOKZiNqyUZWnjIaCIVNQArMHnCZKfEYRg6IM7A+NeJoN8gf/Ws0A== - dependencies: - bn.js "^4.1.0" - elliptic "^6.5.3" - -create-hash@^1.1.0, create-hash@^1.1.2, create-hash@^1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/create-hash/-/create-hash-1.2.0.tgz#889078af11a63756bcfb59bd221996be3a9ef196" - integrity sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg== - dependencies: - cipher-base "^1.0.1" - inherits "^2.0.1" - md5.js "^1.3.4" - ripemd160 "^2.0.1" - sha.js "^2.4.0" - -create-hmac@^1.1.0, create-hmac@^1.1.4, create-hmac@^1.1.7: - version "1.1.7" - resolved "https://registry.yarnpkg.com/create-hmac/-/create-hmac-1.1.7.tgz#69170c78b3ab957147b2b8b04572e47ead2243ff" - integrity sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg== - dependencies: - cipher-base "^1.0.3" - create-hash "^1.1.0" - inherits "^2.0.1" - ripemd160 "^2.0.0" - safe-buffer "^5.0.1" - sha.js "^2.4.8" - -cross-spawn@^6.0.0: - version "6.0.5" - resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-6.0.5.tgz#4a5ec7c64dfae22c3a14124dbacdee846d80cbc4" - integrity sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ== - dependencies: - nice-try "^1.0.4" - path-key "^2.0.1" - semver "^5.5.0" - shebang-command "^1.2.0" - which "^1.2.9" - -crypto-browserify@^3.11.0: - version "3.12.0" - resolved "https://registry.yarnpkg.com/crypto-browserify/-/crypto-browserify-3.12.0.tgz#396cf9f3137f03e4b8e532c58f698254e00f80ec" - integrity sha512-fz4spIh+znjO2VjL+IdhEpRJ3YN6sMzITSBijk6FK2UvTqruSQW+/cCZTSNsMiZNvUeq0CqurF+dAbyiGOY6Wg== - dependencies: - browserify-cipher "^1.0.0" - browserify-sign "^4.0.0" - create-ecdh "^4.0.0" - create-hash "^1.1.0" - create-hmac "^1.1.0" - diffie-hellman "^5.0.0" - inherits "^2.0.1" - pbkdf2 "^3.0.3" - public-encrypt "^4.0.0" - randombytes "^2.0.0" - randomfill "^1.0.3" - -css-color-names@0.0.4, css-color-names@^0.0.4: - version "0.0.4" - resolved "https://registry.yarnpkg.com/css-color-names/-/css-color-names-0.0.4.tgz#808adc2e79cf84738069b646cb20ec27beb629e0" - integrity sha1-gIrcLnnPhHOAabZGyyDsJ762KeA= - -css-declaration-sorter@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/css-declaration-sorter/-/css-declaration-sorter-4.0.1.tgz#c198940f63a76d7e36c1e71018b001721054cb22" - integrity sha512-BcxQSKTSEEQUftYpBVnsH4SF05NTuBokb19/sBt6asXGKZ/6VP7PLG1CBCkFDYOnhXhPh0jMhO6xZ71oYHXHBA== - dependencies: - postcss "^7.0.1" - timsort "^0.3.0" - -css-loader@4.3.0: - version "4.3.0" - resolved "https://registry.yarnpkg.com/css-loader/-/css-loader-4.3.0.tgz#c888af64b2a5b2e85462c72c0f4a85c7e2e0821e" - integrity sha512-rdezjCjScIrsL8BSYszgT4s476IcNKt6yX69t0pHjJVnPUTDpn4WfIpDQTN3wCJvUvfsz/mFjuGOekf3PY3NUg== - dependencies: - camelcase "^6.0.0" - cssesc "^3.0.0" - icss-utils "^4.1.1" - loader-utils "^2.0.0" - postcss "^7.0.32" - postcss-modules-extract-imports "^2.0.0" - postcss-modules-local-by-default "^3.0.3" - postcss-modules-scope "^2.2.0" - postcss-modules-values "^3.0.0" - postcss-value-parser "^4.1.0" - schema-utils "^2.7.1" - semver "^7.3.2" - -css-parse@~2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/css-parse/-/css-parse-2.0.0.tgz#a468ee667c16d81ccf05c58c38d2a97c780dbfd4" - integrity sha1-pGjuZnwW2BzPBcWMONKpfHgNv9Q= - dependencies: - css "^2.0.0" - -css-select-base-adapter@^0.1.1: - version "0.1.1" - resolved "https://registry.yarnpkg.com/css-select-base-adapter/-/css-select-base-adapter-0.1.1.tgz#3b2ff4972cc362ab88561507a95408a1432135d7" - integrity sha512-jQVeeRG70QI08vSTwf1jHxp74JoZsr2XSgETae8/xC8ovSnL2WF87GTLO86Sbwdt2lK4Umg4HnnwMO4YF3Ce7w== - -css-select@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/css-select/-/css-select-2.1.0.tgz#6a34653356635934a81baca68d0255432105dbef" - integrity sha512-Dqk7LQKpwLoH3VovzZnkzegqNSuAziQyNZUcrdDM401iY+R5NkGBXGmtO05/yaXQziALuPogeG0b7UAgjnTJTQ== - dependencies: - boolbase "^1.0.0" - css-what "^3.2.1" - domutils "^1.7.0" - nth-check "^1.0.2" - -css-selector-tokenizer@^0.7.1: - version "0.7.3" - resolved "https://registry.yarnpkg.com/css-selector-tokenizer/-/css-selector-tokenizer-0.7.3.tgz#735f26186e67c749aaf275783405cf0661fae8f1" - integrity sha512-jWQv3oCEL5kMErj4wRnK/OPoBi0D+P1FR2cDCKYPaMeD2eW3/mttav8HT4hT1CKopiJI/psEULjkClhvJo4Lvg== - dependencies: - cssesc "^3.0.0" - fastparse "^1.1.2" - -css-tree@1.0.0-alpha.37: - version "1.0.0-alpha.37" - resolved "https://registry.yarnpkg.com/css-tree/-/css-tree-1.0.0-alpha.37.tgz#98bebd62c4c1d9f960ec340cf9f7522e30709a22" - integrity sha512-DMxWJg0rnz7UgxKT0Q1HU/L9BeJI0M6ksor0OgqOnF+aRCDWg/N2641HmVyU9KVIu0OVVWOb2IpC9A+BJRnejg== - dependencies: - mdn-data "2.0.4" - source-map "^0.6.1" - -css-tree@^1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/css-tree/-/css-tree-1.1.2.tgz#9ae393b5dafd7dae8a622475caec78d3d8fbd7b5" - integrity sha512-wCoWush5Aeo48GLhfHPbmvZs59Z+M7k5+B1xDnXbdWNcEF423DoFdqSWE0PM5aNk5nI5cp1q7ms36zGApY/sKQ== - dependencies: - mdn-data "2.0.14" - source-map "^0.6.1" - -css-what@^3.2.1: - version "3.4.2" - resolved "https://registry.yarnpkg.com/css-what/-/css-what-3.4.2.tgz#ea7026fcb01777edbde52124e21f327e7ae950e4" - integrity sha512-ACUm3L0/jiZTqfzRM3Hi9Q8eZqd6IK37mMWPLz9PJxkLWllYeRf+EHUSHYEtFop2Eqytaq1FizFVh7XfBnXCDQ== - -css@^2.0.0: - version "2.2.4" - resolved "https://registry.yarnpkg.com/css/-/css-2.2.4.tgz#c646755c73971f2bba6a601e2cf2fd71b1298929" - integrity sha512-oUnjmWpy0niI3x/mPL8dVEI1l7MnG3+HHyRPHf+YFSbK+svOhXpmSOcDURUh2aOCgl2grzrOPt1nHLuCVFULLw== - dependencies: - inherits "^2.0.3" - source-map "^0.6.1" - source-map-resolve "^0.5.2" - urix "^0.1.0" - -cssauron@^1.4.0: - version "1.4.0" - resolved "https://registry.yarnpkg.com/cssauron/-/cssauron-1.4.0.tgz#a6602dff7e04a8306dc0db9a551e92e8b5662ad8" - integrity sha1-pmAt/34EqDBtwNuaVR6S6LVmKtg= - dependencies: - through X.X.X - -cssesc@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/cssesc/-/cssesc-3.0.0.tgz#37741919903b868565e1c09ea747445cd18983ee" - integrity sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg== - -cssnano-preset-default@^4.0.7: - version "4.0.7" - resolved "https://registry.yarnpkg.com/cssnano-preset-default/-/cssnano-preset-default-4.0.7.tgz#51ec662ccfca0f88b396dcd9679cdb931be17f76" - integrity sha512-x0YHHx2h6p0fCl1zY9L9roD7rnlltugGu7zXSKQx6k2rYw0Hi3IqxcoAGF7u9Q5w1nt7vK0ulxV8Lo+EvllGsA== - dependencies: - css-declaration-sorter "^4.0.1" - cssnano-util-raw-cache "^4.0.1" - postcss "^7.0.0" - postcss-calc "^7.0.1" - postcss-colormin "^4.0.3" - postcss-convert-values "^4.0.1" - postcss-discard-comments "^4.0.2" - postcss-discard-duplicates "^4.0.2" - postcss-discard-empty "^4.0.1" - postcss-discard-overridden "^4.0.1" - postcss-merge-longhand "^4.0.11" - postcss-merge-rules "^4.0.3" - postcss-minify-font-values "^4.0.2" - postcss-minify-gradients "^4.0.2" - postcss-minify-params "^4.0.2" - postcss-minify-selectors "^4.0.2" - postcss-normalize-charset "^4.0.1" - postcss-normalize-display-values "^4.0.2" - postcss-normalize-positions "^4.0.2" - postcss-normalize-repeat-style "^4.0.2" - postcss-normalize-string "^4.0.2" - postcss-normalize-timing-functions "^4.0.2" - postcss-normalize-unicode "^4.0.1" - postcss-normalize-url "^4.0.1" - postcss-normalize-whitespace "^4.0.2" - postcss-ordered-values "^4.1.2" - postcss-reduce-initial "^4.0.3" - postcss-reduce-transforms "^4.0.2" - postcss-svgo "^4.0.2" - postcss-unique-selectors "^4.0.1" - -cssnano-util-get-arguments@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/cssnano-util-get-arguments/-/cssnano-util-get-arguments-4.0.0.tgz#ed3a08299f21d75741b20f3b81f194ed49cc150f" - integrity sha1-7ToIKZ8h11dBsg87gfGU7UnMFQ8= - -cssnano-util-get-match@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/cssnano-util-get-match/-/cssnano-util-get-match-4.0.0.tgz#c0e4ca07f5386bb17ec5e52250b4f5961365156d" - integrity sha1-wOTKB/U4a7F+xeUiULT1lhNlFW0= - -cssnano-util-raw-cache@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/cssnano-util-raw-cache/-/cssnano-util-raw-cache-4.0.1.tgz#b26d5fd5f72a11dfe7a7846fb4c67260f96bf282" - integrity sha512-qLuYtWK2b2Dy55I8ZX3ky1Z16WYsx544Q0UWViebptpwn/xDBmog2TLg4f+DBMg1rJ6JDWtn96WHbOKDWt1WQA== - dependencies: - postcss "^7.0.0" - -cssnano-util-same-parent@^4.0.0: - version "4.0.1" - resolved "https://registry.yarnpkg.com/cssnano-util-same-parent/-/cssnano-util-same-parent-4.0.1.tgz#574082fb2859d2db433855835d9a8456ea18bbf3" - integrity sha512-WcKx5OY+KoSIAxBW6UBBRay1U6vkYheCdjyVNDm85zt5K9mHoGOfsOsqIszfAqrQQFIIKgjh2+FDgIj/zsl21Q== - -cssnano@4.1.10: - version "4.1.10" - resolved "https://registry.yarnpkg.com/cssnano/-/cssnano-4.1.10.tgz#0ac41f0b13d13d465487e111b778d42da631b8b2" - integrity sha512-5wny+F6H4/8RgNlaqab4ktc3e0/blKutmq8yNlBFXA//nSFFAqAngjNVRzUvCgYROULmZZUoosL/KSoZo5aUaQ== - dependencies: - cosmiconfig "^5.0.0" - cssnano-preset-default "^4.0.7" - is-resolvable "^1.0.0" - postcss "^7.0.0" - -csso@^4.0.2: - version "4.2.0" - resolved "https://registry.yarnpkg.com/csso/-/csso-4.2.0.tgz#ea3a561346e8dc9f546d6febedd50187cf389529" - integrity sha512-wvlcdIbf6pwKEk7vHj8/Bkc0B4ylXZruLvOgs9doS5eOsOpuodOV2zJChSpkp+pRpYQLQMeF04nr3Z68Sta9jA== - dependencies: - css-tree "^1.1.2" - -custom-event@~1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/custom-event/-/custom-event-1.0.1.tgz#5d02a46850adf1b4a317946a3928fccb5bfd0425" - integrity sha1-XQKkaFCt8bSjF5RqOSj8y1v9BCU= - -cyclist@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/cyclist/-/cyclist-1.0.1.tgz#596e9698fd0c80e12038c2b82d6eb1b35b6224d9" - integrity sha1-WW6WmP0MgOEgOMK4LW6xs1tiJNk= - -d@1, d@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/d/-/d-1.0.1.tgz#8698095372d58dbee346ffd0c7093f99f8f9eb5a" - integrity sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA== - dependencies: - es5-ext "^0.10.50" - type "^1.0.1" - -damerau-levenshtein@^1.0.4: - version "1.0.6" - resolved "https://registry.yarnpkg.com/damerau-levenshtein/-/damerau-levenshtein-1.0.6.tgz#143c1641cb3d85c60c32329e26899adea8701791" - integrity sha512-JVrozIeElnj3QzfUIt8tB8YMluBJom4Vw9qTPpjGYQ9fYlB3D/rb6OordUxf3xeFB35LKWs0xqcO5U6ySvBtug== - -dashdash@^1.12.0: - version "1.14.1" - resolved "https://registry.yarnpkg.com/dashdash/-/dashdash-1.14.1.tgz#853cfa0f7cbe2fed5de20326b8dd581035f6e2f0" - integrity sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA= - dependencies: - assert-plus "^1.0.0" - -date-format@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/date-format/-/date-format-2.1.0.tgz#31d5b5ea211cf5fd764cd38baf9d033df7e125cf" - integrity sha512-bYQuGLeFxhkxNOF3rcMtiZxvCBAquGzZm6oWA1oZ0g2THUzivaRhv8uOhdr19LmoobSOLoIAxeUK2RdbM8IFTA== - -date-format@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/date-format/-/date-format-3.0.0.tgz#eb8780365c7d2b1511078fb491e6479780f3ad95" - integrity sha512-eyTcpKOcamdhWJXj56DpQMo1ylSQpcGtGKXcU0Tb97+K56/CF5amAqqqNj0+KvA0iw2ynxtHWFsPDSClCxe48w== - -debug@2.6.9, debug@^2.2.0, debug@^2.3.3, debug@^2.6.9: - version "2.6.9" - resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f" - integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA== - dependencies: - ms "2.0.0" - -debug@3.1.0, debug@~3.1.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/debug/-/debug-3.1.0.tgz#5bb5a0672628b64149566ba16819e61518c67261" - integrity sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g== - dependencies: - ms "2.0.0" - -debug@4.2.0: - version "4.2.0" - resolved "https://registry.yarnpkg.com/debug/-/debug-4.2.0.tgz#7f150f93920e94c58f5574c2fd01a3110effe7f1" - integrity sha512-IX2ncY78vDTjZMFUdmsvIRFY2Cf4FnD0wRs+nQwJU8Lu99/tPFdb0VybiiMTPe3I6rQmwsqQqRBvxU+bZ/I8sg== - dependencies: - ms "2.1.2" - -debug@^3.1.0, debug@^3.1.1, debug@^3.2.5: - version "3.2.7" - resolved "https://registry.yarnpkg.com/debug/-/debug-3.2.7.tgz#72580b7e9145fb39b6676f9c5e5fb100b934179a" - integrity sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ== - dependencies: - ms "^2.1.1" - -debug@^4.1.0, debug@^4.1.1: - version "4.3.1" - resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.1.tgz#f0d229c505e0c6d8c49ac553d1b13dc183f6b2ee" - integrity sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ== - dependencies: - ms "2.1.2" - -debug@~4.1.0: - version "4.1.1" - resolved "https://registry.yarnpkg.com/debug/-/debug-4.1.1.tgz#3b72260255109c6b589cee050f1d516139664791" - integrity sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw== - dependencies: - ms "^2.1.1" - -decamelize@^1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-1.2.0.tgz#f6534d15148269b20352e7bee26f501f9a191290" - integrity sha1-9lNNFRSCabIDUue+4m9QH5oZEpA= - -decode-uri-component@^0.2.0: - version "0.2.0" - resolved "https://registry.yarnpkg.com/decode-uri-component/-/decode-uri-component-0.2.0.tgz#eb3913333458775cb84cd1a1fae062106bb87545" - integrity sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU= - -decompress-response@^3.3.0: - version "3.3.0" - resolved "https://registry.yarnpkg.com/decompress-response/-/decompress-response-3.3.0.tgz#80a4dd323748384bfa248083622aedec982adff3" - integrity sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M= - dependencies: - mimic-response "^1.0.0" - -deep-equal@^1.0.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/deep-equal/-/deep-equal-1.1.1.tgz#b5c98c942ceffaf7cb051e24e1434a25a2e6076a" - integrity sha512-yd9c5AdiqVcR+JjcwUQb9DkhJc8ngNr0MahEBGvDiJw8puWab2yZlh+nkasOnZP+EGTAP6rRp2JzJhJZzvNF8g== - dependencies: - is-arguments "^1.0.4" - is-date-object "^1.0.1" - is-regex "^1.0.4" - object-is "^1.0.1" - object-keys "^1.1.1" - regexp.prototype.flags "^1.2.0" - -default-gateway@^4.2.0: - version "4.2.0" - resolved "https://registry.yarnpkg.com/default-gateway/-/default-gateway-4.2.0.tgz#167104c7500c2115f6dd69b0a536bb8ed720552b" - integrity sha512-h6sMrVB1VMWVrW13mSc6ia/DwYYw5MN6+exNu1OaJeFac5aSAvwM7lZ0NVfTABuSkQelr4h5oebg3KB1XPdjgA== - dependencies: - execa "^1.0.0" - ip-regex "^2.1.0" - -defaults@^1.0.3: - version "1.0.3" - resolved "https://registry.yarnpkg.com/defaults/-/defaults-1.0.3.tgz#c656051e9817d9ff08ed881477f3fe4019f3ef7d" - integrity sha1-xlYFHpgX2f8I7YgUd/P+QBnz730= - dependencies: - clone "^1.0.2" - -defer-to-connect@^1.0.1: - version "1.1.3" - resolved "https://registry.yarnpkg.com/defer-to-connect/-/defer-to-connect-1.1.3.tgz#331ae050c08dcf789f8c83a7b81f0ed94f4ac591" - integrity sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ== - -define-properties@^1.1.3: - version "1.1.3" - resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.1.3.tgz#cf88da6cbee26fe6db7094f61d870cbd84cee9f1" - integrity sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ== - dependencies: - object-keys "^1.0.12" - -define-property@^0.2.5: - version "0.2.5" - resolved "https://registry.yarnpkg.com/define-property/-/define-property-0.2.5.tgz#c35b1ef918ec3c990f9a5bc57be04aacec5c8116" - integrity sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY= - dependencies: - is-descriptor "^0.1.0" - -define-property@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/define-property/-/define-property-1.0.0.tgz#769ebaaf3f4a63aad3af9e8d304c9bbe79bfb0e6" - integrity sha1-dp66rz9KY6rTr56NMEybvnm/sOY= - dependencies: - is-descriptor "^1.0.0" - -define-property@^2.0.2: - version "2.0.2" - resolved "https://registry.yarnpkg.com/define-property/-/define-property-2.0.2.tgz#d459689e8d654ba77e02a817f8710d702cb16e9d" - integrity sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ== - dependencies: - is-descriptor "^1.0.2" - isobject "^3.0.1" - -del@^2.2.0: - version "2.2.2" - resolved "https://registry.yarnpkg.com/del/-/del-2.2.2.tgz#c12c981d067846c84bcaf862cff930d907ffd1a8" - integrity sha1-wSyYHQZ4RshLyvhiz/kw2Qf/0ag= - dependencies: - globby "^5.0.0" - is-path-cwd "^1.0.0" - is-path-in-cwd "^1.0.0" - object-assign "^4.0.1" - pify "^2.0.0" - pinkie-promise "^2.0.0" - rimraf "^2.2.8" - -del@^4.1.1: - version "4.1.1" - resolved "https://registry.yarnpkg.com/del/-/del-4.1.1.tgz#9e8f117222ea44a31ff3a156c049b99052a9f0b4" - integrity sha512-QwGuEUouP2kVwQenAsOof5Fv8K9t3D8Ca8NxcXKrIpEHjTXK5J2nXLdP+ALI1cgv8wj7KuwBhTwBkOZSJKM5XQ== - dependencies: - "@types/glob" "^7.1.1" - globby "^6.1.0" - is-path-cwd "^2.0.0" - is-path-in-cwd "^2.0.0" - p-map "^2.0.0" - pify "^4.0.1" - rimraf "^2.6.3" - -delayed-stream@~1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619" - integrity sha1-3zrhmayt+31ECqrgsp4icrJOxhk= - -depd@~1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/depd/-/depd-1.1.2.tgz#9bcd52e14c097763e749b274c4346ed2e560b5a9" - integrity sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak= - -dependency-graph@^0.7.2: - version "0.7.2" - resolved "https://registry.yarnpkg.com/dependency-graph/-/dependency-graph-0.7.2.tgz#91db9de6eb72699209d88aea4c1fd5221cac1c49" - integrity sha512-KqtH4/EZdtdfWX0p6MGP9jljvxSY6msy/pRUD4jgNwVpv3v1QmNLlsB3LDSSUg79BRVSn7jI1QPRtArGABovAQ== - -des.js@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/des.js/-/des.js-1.0.1.tgz#5382142e1bdc53f85d86d53e5f4aa7deb91e0843" - integrity sha512-Q0I4pfFrv2VPd34/vfLrFOoRmlYj3OV50i7fskps1jZWK1kApMWWT9G6RRUeYedLcBDIhnSDaUvJMb3AhUlaEA== - dependencies: - inherits "^2.0.1" - minimalistic-assert "^1.0.0" - -destroy@~1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.0.4.tgz#978857442c44749e4206613e37946205826abd80" - integrity sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA= - -detect-node@^2.0.4: - version "2.0.4" - resolved "https://registry.yarnpkg.com/detect-node/-/detect-node-2.0.4.tgz#014ee8f8f669c5c58023da64b8179c083a28c46c" - integrity sha512-ZIzRpLJrOj7jjP2miAtgqIfmzbxa4ZOr5jJc601zklsfEx9oTzmmj2nVpIPRpNlRTIh8lc1kyViIY7BWSGNmKw== - -di@^0.0.1: - version "0.0.1" - resolved "https://registry.yarnpkg.com/di/-/di-0.0.1.tgz#806649326ceaa7caa3306d75d985ea2748ba913c" - integrity sha1-gGZJMmzqp8qjMG112YXqJ0i6kTw= - -diff@^4.0.1: - version "4.0.2" - resolved "https://registry.yarnpkg.com/diff/-/diff-4.0.2.tgz#60f3aecb89d5fae520c11aa19efc2bb982aade7d" - integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A== - -diffie-hellman@^5.0.0: - version "5.0.3" - resolved "https://registry.yarnpkg.com/diffie-hellman/-/diffie-hellman-5.0.3.tgz#40e8ee98f55a2149607146921c63e1ae5f3d2875" - integrity sha512-kqag/Nl+f3GwyK25fhUMYj81BUOrZ9IuJsjIcDE5icNM9FJHAVm3VcUDxdLPoQtTuUylWm6ZIknYJwwaPxsUzg== - dependencies: - bn.js "^4.1.0" - miller-rabin "^4.0.0" - randombytes "^2.0.0" - -dir-glob@^3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/dir-glob/-/dir-glob-3.0.1.tgz#56dbf73d992a4a93ba1584f4534063fd2e41717f" - integrity sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA== - dependencies: - path-type "^4.0.0" - -dns-equal@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/dns-equal/-/dns-equal-1.0.0.tgz#b39e7f1da6eb0a75ba9c17324b34753c47e0654d" - integrity sha1-s55/HabrCnW6nBcySzR1PEfgZU0= - -dns-packet@^1.3.1: - version "1.3.1" - resolved "https://registry.yarnpkg.com/dns-packet/-/dns-packet-1.3.1.tgz#12aa426981075be500b910eedcd0b47dd7deda5a" - integrity sha512-0UxfQkMhYAUaZI+xrNZOz/as5KgDU0M/fQ9b6SpkyLbk3GEswDi6PADJVaYJradtRVsRIlF1zLyOodbcTCDzUg== - dependencies: - ip "^1.1.0" - safe-buffer "^5.0.1" - -dns-txt@^2.0.2: - version "2.0.2" - resolved "https://registry.yarnpkg.com/dns-txt/-/dns-txt-2.0.2.tgz#b91d806f5d27188e4ab3e7d107d881a1cc4642b6" - integrity sha1-uR2Ab10nGI5Ks+fRB9iBocxGQrY= - dependencies: - buffer-indexof "^1.0.0" - -dom-serialize@^2.2.1: - version "2.2.1" - resolved "https://registry.yarnpkg.com/dom-serialize/-/dom-serialize-2.2.1.tgz#562ae8999f44be5ea3076f5419dcd59eb43ac95b" - integrity sha1-ViromZ9Evl6jB29UGdzVnrQ6yVs= - dependencies: - custom-event "~1.0.0" - ent "~2.2.0" - extend "^3.0.0" - void-elements "^2.0.0" - -dom-serializer@0: - version "0.2.2" - resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-0.2.2.tgz#1afb81f533717175d478655debc5e332d9f9bb51" - integrity sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g== - dependencies: - domelementtype "^2.0.1" - entities "^2.0.0" - -domain-browser@^1.1.1: - version "1.2.0" - resolved "https://registry.yarnpkg.com/domain-browser/-/domain-browser-1.2.0.tgz#3d31f50191a6749dd1375a7f522e823d42e54eda" - integrity sha512-jnjyiM6eRyZl2H+W8Q/zLMA481hzi0eszAaBUzIVnmYVDBbnLxVNnfu1HgEBvCbL+71FrxMl3E6lpKH7Ge3OXA== - -domelementtype@1: - version "1.3.1" - resolved "https://registry.yarnpkg.com/domelementtype/-/domelementtype-1.3.1.tgz#d048c44b37b0d10a7f2a3d5fee3f4333d790481f" - integrity sha512-BSKB+TSpMpFI/HOxCNr1O8aMOTZ8hT3pM3GQ0w/mWRmkhEDSFJkkyzz4XQsBV44BChwGkrDfMyjVD0eA2aFV3w== - -domelementtype@^2.0.1: - version "2.1.0" - resolved "https://registry.yarnpkg.com/domelementtype/-/domelementtype-2.1.0.tgz#a851c080a6d1c3d94344aed151d99f669edf585e" - integrity sha512-LsTgx/L5VpD+Q8lmsXSHW2WpA+eBlZ9HPf3erD1IoPF00/3JKHZ3BknUVA2QGDNu69ZNmyFmCWBSO45XjYKC5w== - -domutils@^1.7.0: - version "1.7.0" - resolved "https://registry.yarnpkg.com/domutils/-/domutils-1.7.0.tgz#56ea341e834e06e6748af7a1cb25da67ea9f8c2a" - integrity sha512-Lgd2XcJ/NjEw+7tFvfKxOzCYKZsdct5lczQ2ZaQY8Djz7pfAD3Gbp8ySJWtreII/vDlMVmxwa6pHmdxIYgttDg== - dependencies: - dom-serializer "0" - domelementtype "1" - -dot-prop@^5.2.0: - version "5.3.0" - resolved "https://registry.yarnpkg.com/dot-prop/-/dot-prop-5.3.0.tgz#90ccce708cd9cd82cc4dc8c3ddd9abdd55b20e88" - integrity sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q== - dependencies: - is-obj "^2.0.0" - -duplexer3@^0.1.4: - version "0.1.4" - resolved "https://registry.yarnpkg.com/duplexer3/-/duplexer3-0.1.4.tgz#ee01dd1cac0ed3cbc7fdbea37dc0a8f1ce002ce2" - integrity sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI= - -duplexify@^3.4.2, duplexify@^3.6.0: - version "3.7.1" - resolved "https://registry.yarnpkg.com/duplexify/-/duplexify-3.7.1.tgz#2a4df5317f6ccfd91f86d6fd25d8d8a103b88309" - integrity sha512-07z8uv2wMyS51kKhD1KsdXJg5WQ6t93RneqRxUHnskXVtlYYkLqM0gqStQZ3pj073g687jPCHrqNfCzawLYh5g== - dependencies: - end-of-stream "^1.0.0" - inherits "^2.0.1" - readable-stream "^2.0.0" - stream-shift "^1.0.0" - -ecc-jsbn@~0.1.1: - version "0.1.2" - resolved "https://registry.yarnpkg.com/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz#3a83a904e54353287874c564b7549386849a98c9" - integrity sha1-OoOpBOVDUyh4dMVkt1SThoSamMk= - dependencies: - jsbn "~0.1.0" - safer-buffer "^2.1.0" - -ee-first@1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d" - integrity sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0= - -electron-to-chromium@^1.3.649: - version "1.3.687" - resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.3.687.tgz#c336184b7ab70427ffe2ee79eaeaedbc1ad8c374" - integrity sha512-IpzksdQNl3wdgkzf7dnA7/v10w0Utf1dF2L+B4+gKrloBrxCut+au+kky3PYvle3RMdSxZP+UiCZtLbcYRxSNQ== - -electron@11.1.1: - version "11.1.1" - resolved "https://registry.yarnpkg.com/electron/-/electron-11.1.1.tgz#188f036f8282798398dca9513e9bb3b10213e3aa" - integrity sha512-tlbex3xosJgfileN6BAQRotevPRXB/wQIq48QeQ08tUJJrXwE72c8smsM/hbHx5eDgnbfJ2G3a60PmRjHU2NhA== - dependencies: - "@electron/get" "^1.0.1" - "@types/node" "^12.0.12" - extract-zip "^1.0.3" - -elliptic@^6.5.3: - version "6.5.4" - resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.5.4.tgz#da37cebd31e79a1367e941b592ed1fbebd58abbb" - integrity sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ== - dependencies: - bn.js "^4.11.9" - brorand "^1.1.0" - hash.js "^1.0.0" - hmac-drbg "^1.0.1" - inherits "^2.0.4" - minimalistic-assert "^1.0.1" - minimalistic-crypto-utils "^1.0.1" - -emoji-regex@^7.0.1: - version "7.0.3" - resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-7.0.3.tgz#933a04052860c85e83c122479c4748a8e4c72156" - integrity sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA== - -emoji-regex@^8.0.0: - version "8.0.0" - resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37" - integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A== - -emojis-list@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/emojis-list/-/emojis-list-2.1.0.tgz#4daa4d9db00f9819880c79fa457ae5b09a1fd389" - integrity sha1-TapNnbAPmBmIDHn6RXrlsJof04k= - -emojis-list@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/emojis-list/-/emojis-list-3.0.0.tgz#5570662046ad29e2e916e71aae260abdff4f6a78" - integrity sha512-/kyM18EfinwXZbno9FyUGeFh87KC8HRQBQGildHZbEuRyWFOmv1U10o9BBp8XVZDVNNuQKyIGIu5ZYAAXJ0V2Q== - -encodeurl@^1.0.2, encodeurl@~1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59" - integrity sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k= - -encoding@^0.1.11: - version "0.1.13" - resolved "https://registry.yarnpkg.com/encoding/-/encoding-0.1.13.tgz#56574afdd791f54a8e9b2785c0582a2d26210fa9" - integrity sha512-ETBauow1T35Y/WZMkio9jiM0Z5xjHHmJ4XmjZOq1l/dXz3lr2sRn87nJy20RupqSh1F2m3HHPSp8ShIPQJrJ3A== - dependencies: - iconv-lite "^0.6.2" - -end-of-stream@^1.0.0, end-of-stream@^1.1.0: - version "1.4.4" - resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.4.tgz#5ae64a5f45057baf3626ec14da0ca5e4b2431eb0" - integrity sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q== - dependencies: - once "^1.4.0" - -engine.io-client@~3.5.0: - version "3.5.1" - resolved "https://registry.yarnpkg.com/engine.io-client/-/engine.io-client-3.5.1.tgz#b500458a39c0cd197a921e0e759721a746d0bdb9" - integrity sha512-oVu9kBkGbcggulyVF0kz6BV3ganqUeqXvD79WOFKa+11oK692w1NyFkuEj4xrkFRpZhn92QOqTk4RQq5LiBXbQ== - dependencies: - component-emitter "~1.3.0" - component-inherit "0.0.3" - debug "~3.1.0" - engine.io-parser "~2.2.0" - has-cors "1.1.0" - indexof "0.0.1" - parseqs "0.0.6" - parseuri "0.0.6" - ws "~7.4.2" - xmlhttprequest-ssl "~1.5.4" - yeast "0.1.2" - -engine.io-parser@~2.2.0: - version "2.2.1" - resolved "https://registry.yarnpkg.com/engine.io-parser/-/engine.io-parser-2.2.1.tgz#57ce5611d9370ee94f99641b589f94c97e4f5da7" - integrity sha512-x+dN/fBH8Ro8TFwJ+rkB2AmuVw9Yu2mockR/p3W8f8YtExwFgDvBDi0GWyb4ZLkpahtDGZgtr3zLovanJghPqg== - dependencies: - after "0.8.2" - arraybuffer.slice "~0.0.7" - base64-arraybuffer "0.1.4" - blob "0.0.5" - has-binary2 "~1.0.2" - -engine.io@~3.5.0: - version "3.5.0" - resolved "https://registry.yarnpkg.com/engine.io/-/engine.io-3.5.0.tgz#9d6b985c8a39b1fe87cd91eb014de0552259821b" - integrity sha512-21HlvPUKaitDGE4GXNtQ7PLP0Sz4aWLddMPw2VTyFz1FVZqu/kZsJUO8WNpKuE/OCL7nkfRaOui2ZCJloGznGA== - dependencies: - accepts "~1.3.4" - base64id "2.0.0" - cookie "~0.4.1" - debug "~4.1.0" - engine.io-parser "~2.2.0" - ws "~7.4.2" - -enhanced-resolve@5.3.1: - version "5.3.1" - resolved "https://registry.yarnpkg.com/enhanced-resolve/-/enhanced-resolve-5.3.1.tgz#3f988d0d7775bdc2d96ede321dc81f8249492f57" - integrity sha512-G1XD3MRGrGfNcf6Hg0LVZG7GIKcYkbfHa5QMxt1HDUTdYoXH0JR1xXyg+MaKLF73E9A27uWNVxvFivNRYeUB6w== - dependencies: - graceful-fs "^4.2.4" - tapable "^2.0.0" - -enhanced-resolve@^4.3.0: - version "4.5.0" - resolved "https://registry.yarnpkg.com/enhanced-resolve/-/enhanced-resolve-4.5.0.tgz#2f3cfd84dbe3b487f18f2db2ef1e064a571ca5ec" - integrity sha512-Nv9m36S/vxpsI+Hc4/ZGRs0n9mXqSWGGq49zxb/cJfPAQMbUtttJAlNPS4AQzaBdw/pKskw5bMbekT/Y7W/Wlg== - dependencies: - graceful-fs "^4.1.2" - memory-fs "^0.5.0" - tapable "^1.0.0" - -ent@~2.2.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/ent/-/ent-2.2.0.tgz#e964219325a21d05f44466a2f686ed6ce5f5dd1d" - integrity sha1-6WQhkyWiHQX0RGai9obtbOX13R0= - -entities@^2.0.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/entities/-/entities-2.2.0.tgz#098dc90ebb83d8dffa089d55256b351d34c4da55" - integrity sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A== - -env-paths@^2.2.0: - version "2.2.1" - resolved "https://registry.yarnpkg.com/env-paths/-/env-paths-2.2.1.tgz#420399d416ce1fbe9bc0a07c62fa68d67fd0f8f2" - integrity sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A== - -err-code@^1.0.0: - version "1.1.2" - resolved "https://registry.yarnpkg.com/err-code/-/err-code-1.1.2.tgz#06e0116d3028f6aef4806849eb0ea6a748ae6960" - integrity sha1-BuARbTAo9q70gGhJ6w6mp0iuaWA= - -errno@^0.1.1, errno@^0.1.3, errno@~0.1.7: - version "0.1.8" - resolved "https://registry.yarnpkg.com/errno/-/errno-0.1.8.tgz#8bb3e9c7d463be4976ff888f76b4809ebc2e811f" - integrity sha512-dJ6oBr5SQ1VSd9qkk7ByRgb/1SH4JZjCHSW/mr63/QcXO9zLVxvJ6Oy13nio03rxpSnVDDjFor75SjVeZWPW/A== - dependencies: - prr "~1.0.1" - -error-ex@^1.3.1: - version "1.3.2" - resolved "https://registry.yarnpkg.com/error-ex/-/error-ex-1.3.2.tgz#b4ac40648107fdcdcfae242f428bea8a14d4f1bf" - integrity sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g== - dependencies: - is-arrayish "^0.2.1" - -es-abstract@^1.17.2, es-abstract@^1.18.0-next.2: - version "1.18.0" - resolved "https://registry.yarnpkg.com/es-abstract/-/es-abstract-1.18.0.tgz#ab80b359eecb7ede4c298000390bc5ac3ec7b5a4" - integrity sha512-LJzK7MrQa8TS0ja2w3YNLzUgJCGPdPOV1yVvezjNnS89D+VR08+Szt2mz3YB2Dck/+w5tfIq/RoUAFqJJGM2yw== - dependencies: - call-bind "^1.0.2" - es-to-primitive "^1.2.1" - function-bind "^1.1.1" - get-intrinsic "^1.1.1" - has "^1.0.3" - has-symbols "^1.0.2" - is-callable "^1.2.3" - is-negative-zero "^2.0.1" - is-regex "^1.1.2" - is-string "^1.0.5" - object-inspect "^1.9.0" - object-keys "^1.1.1" - object.assign "^4.1.2" - string.prototype.trimend "^1.0.4" - string.prototype.trimstart "^1.0.4" - unbox-primitive "^1.0.0" - -es-to-primitive@^1.2.1: - version "1.2.1" - resolved "https://registry.yarnpkg.com/es-to-primitive/-/es-to-primitive-1.2.1.tgz#e55cd4c9cdc188bcefb03b366c736323fc5c898a" - integrity sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA== - dependencies: - is-callable "^1.1.4" - is-date-object "^1.0.1" - is-symbol "^1.0.2" - -es5-ext@^0.10.35, es5-ext@^0.10.50: - version "0.10.53" - resolved "https://registry.yarnpkg.com/es5-ext/-/es5-ext-0.10.53.tgz#93c5a3acfdbef275220ad72644ad02ee18368de1" - integrity sha512-Xs2Stw6NiNHWypzRTY1MtaG/uJlwCk8kH81920ma8mvN8Xq1gsfhZvpkImLQArw8AHnv8MT2I45J3c0R8slE+Q== - dependencies: - es6-iterator "~2.0.3" - es6-symbol "~3.1.3" - next-tick "~1.0.0" - -es6-error@^4.1.1: - version "4.1.1" - resolved "https://registry.yarnpkg.com/es6-error/-/es6-error-4.1.1.tgz#9e3af407459deed47e9a91f9b885a84eb05c561d" - integrity sha512-Um/+FxMr9CISWh0bi5Zv0iOD+4cFh5qLeks1qhAopKVAJw3drgKbKySikp7wGhDL0HPeaja0P5ULZrxLkniUVg== - -es6-iterator@2.0.3, es6-iterator@~2.0.3: - version "2.0.3" - resolved "https://registry.yarnpkg.com/es6-iterator/-/es6-iterator-2.0.3.tgz#a7de889141a05a94b0854403b2d0a0fbfa98f3b7" - integrity sha1-p96IkUGgWpSwhUQDstCg+/qY87c= - dependencies: - d "1" - es5-ext "^0.10.35" - es6-symbol "^3.1.1" - -es6-promise@^4.0.3: - version "4.2.8" - resolved "https://registry.yarnpkg.com/es6-promise/-/es6-promise-4.2.8.tgz#4eb21594c972bc40553d276e510539143db53e0a" - integrity sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w== - -es6-promisify@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/es6-promisify/-/es6-promisify-5.0.0.tgz#5109d62f3e56ea967c4b63505aef08291c8a5203" - integrity sha1-UQnWLz5W6pZ8S2NQWu8IKRyKUgM= - dependencies: - es6-promise "^4.0.3" - -es6-symbol@^3.1.1, es6-symbol@~3.1.3: - version "3.1.3" - resolved "https://registry.yarnpkg.com/es6-symbol/-/es6-symbol-3.1.3.tgz#bad5d3c1bcdac28269f4cb331e431c78ac705d18" - integrity sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA== - dependencies: - d "^1.0.1" - ext "^1.1.2" - -escalade@^3.1.1: - version "3.1.1" - resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.1.1.tgz#d8cfdc7000965c5a0174b4a82eaa5c0552742e40" - integrity sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw== - -escape-html@~1.0.3: - version "1.0.3" - resolved "https://registry.yarnpkg.com/escape-html/-/escape-html-1.0.3.tgz#0258eae4d3d0c0974de1c169188ef0051d1d1988" - integrity sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg= - -escape-string-regexp@^1.0.2, escape-string-regexp@^1.0.5: - version "1.0.5" - resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4" - integrity sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ= - -escape-string-regexp@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34" - integrity sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA== - -eslint-scope@^4.0.3: - version "4.0.3" - resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-4.0.3.tgz#ca03833310f6889a3264781aa82e63eb9cfe7848" - integrity sha512-p7VutNr1O/QrxysMo3E45FjYDTeXBy0iTltPFNSqKAIfjDSXC+4dj+qfyuD8bfAXrW/y6lW3O76VaYNPKfpKrg== - dependencies: - esrecurse "^4.1.0" - estraverse "^4.1.1" - -esprima@^4.0.0: - version "4.0.1" - resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz#13b04cdb3e6c5d19df91ab6987a8695619b0aa71" - integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A== - -esrecurse@^4.1.0: - version "4.3.0" - resolved "https://registry.yarnpkg.com/esrecurse/-/esrecurse-4.3.0.tgz#7ad7964d679abb28bee72cec63758b1c5d2c9921" - integrity sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag== - dependencies: - estraverse "^5.2.0" - -estraverse@^4.1.1: - version "4.3.0" - resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-4.3.0.tgz#398ad3f3c5a24948be7725e83d11a7de28cdbd1d" - integrity sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw== - -estraverse@^5.2.0: - version "5.2.0" - resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-5.2.0.tgz#307df42547e6cc7324d3cf03c155d5cdb8c53880" - integrity sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ== - -esutils@^2.0.2: - version "2.0.3" - resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64" - integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g== - -etag@~1.8.1: - version "1.8.1" - resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887" - integrity sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc= - -eventemitter3@^4.0.0: - version "4.0.7" - resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-4.0.7.tgz#2de9b68f6528d5644ef5c59526a1b4a07306169f" - integrity sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw== - -events@^3.0.0: - version "3.3.0" - resolved "https://registry.yarnpkg.com/events/-/events-3.3.0.tgz#31a95ad0a924e2d2c419a813aeb2c4e878ea7400" - integrity sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q== - -eventsource@^1.0.7: - version "1.0.7" - resolved "https://registry.yarnpkg.com/eventsource/-/eventsource-1.0.7.tgz#8fbc72c93fcd34088090bc0a4e64f4b5cee6d8d0" - integrity sha512-4Ln17+vVT0k8aWq+t/bF5arcS3EpT9gYtW66EPacdj/mAFevznsnyoHLPy2BA8gbIQeIHoPsvwmfBftfcG//BQ== - dependencies: - original "^1.0.0" - -evp_bytestokey@^1.0.0, evp_bytestokey@^1.0.3: - version "1.0.3" - resolved "https://registry.yarnpkg.com/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz#7fcbdb198dc71959432efe13842684e0525acb02" - integrity sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA== - dependencies: - md5.js "^1.3.4" - safe-buffer "^5.1.1" - -execa@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/execa/-/execa-1.0.0.tgz#c6236a5bb4df6d6f15e88e7f017798216749ddd8" - integrity sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA== - dependencies: - cross-spawn "^6.0.0" - get-stream "^4.0.0" - is-stream "^1.1.0" - npm-run-path "^2.0.0" - p-finally "^1.0.0" - signal-exit "^3.0.0" - strip-eof "^1.0.0" - -exit@^0.1.2: - version "0.1.2" - resolved "https://registry.yarnpkg.com/exit/-/exit-0.1.2.tgz#0632638f8d877cc82107d30a0fff1a17cba1cd0c" - integrity sha1-BjJjj42HfMghB9MKD/8aF8uhzQw= - -expand-brackets@^2.1.4: - version "2.1.4" - resolved "https://registry.yarnpkg.com/expand-brackets/-/expand-brackets-2.1.4.tgz#b77735e315ce30f6b6eff0f83b04151a22449622" - integrity sha1-t3c14xXOMPa27/D4OwQVGiJEliI= - dependencies: - debug "^2.3.3" - define-property "^0.2.5" - extend-shallow "^2.0.1" - posix-character-classes "^0.1.0" - regex-not "^1.0.0" - snapdragon "^0.8.1" - to-regex "^3.0.1" - -express@^4.17.1: - version "4.17.1" - resolved "https://registry.yarnpkg.com/express/-/express-4.17.1.tgz#4491fc38605cf51f8629d39c2b5d026f98a4c134" - integrity sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g== - dependencies: - accepts "~1.3.7" - array-flatten "1.1.1" - body-parser "1.19.0" - content-disposition "0.5.3" - content-type "~1.0.4" - cookie "0.4.0" - cookie-signature "1.0.6" - debug "2.6.9" - depd "~1.1.2" - encodeurl "~1.0.2" - escape-html "~1.0.3" - etag "~1.8.1" - finalhandler "~1.1.2" - fresh "0.5.2" - merge-descriptors "1.0.1" - methods "~1.1.2" - on-finished "~2.3.0" - parseurl "~1.3.3" - path-to-regexp "0.1.7" - proxy-addr "~2.0.5" - qs "6.7.0" - range-parser "~1.2.1" - safe-buffer "5.1.2" - send "0.17.1" - serve-static "1.14.1" - setprototypeof "1.1.1" - statuses "~1.5.0" - type-is "~1.6.18" - utils-merge "1.0.1" - vary "~1.1.2" - -ext@^1.1.2: - version "1.4.0" - resolved "https://registry.yarnpkg.com/ext/-/ext-1.4.0.tgz#89ae7a07158f79d35517882904324077e4379244" - integrity sha512-Key5NIsUxdqKg3vIsdw9dSuXpPCQ297y6wBjL30edxwPgt2E44WcWBZey/ZvUc6sERLTxKdyCu4gZFmUbk1Q7A== - dependencies: - type "^2.0.0" - -extend-shallow@^2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/extend-shallow/-/extend-shallow-2.0.1.tgz#51af7d614ad9a9f610ea1bafbb989d6b1c56890f" - integrity sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8= - dependencies: - is-extendable "^0.1.0" - -extend-shallow@^3.0.0, extend-shallow@^3.0.2: - version "3.0.2" - resolved "https://registry.yarnpkg.com/extend-shallow/-/extend-shallow-3.0.2.tgz#26a71aaf073b39fb2127172746131c2704028db8" - integrity sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg= - dependencies: - assign-symbols "^1.0.0" - is-extendable "^1.0.1" - -extend@^3.0.0, extend@~3.0.2: - version "3.0.2" - resolved "https://registry.yarnpkg.com/extend/-/extend-3.0.2.tgz#f8b1136b4071fbd8eb140aff858b1019ec2915fa" - integrity sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g== - -external-editor@^3.0.3: - version "3.1.0" - resolved "https://registry.yarnpkg.com/external-editor/-/external-editor-3.1.0.tgz#cb03f740befae03ea4d283caed2741a83f335495" - integrity sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew== - dependencies: - chardet "^0.7.0" - iconv-lite "^0.4.24" - tmp "^0.0.33" - -extglob@^2.0.4: - version "2.0.4" - resolved "https://registry.yarnpkg.com/extglob/-/extglob-2.0.4.tgz#ad00fe4dc612a9232e8718711dc5cb5ab0285543" - integrity sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw== - dependencies: - array-unique "^0.3.2" - define-property "^1.0.0" - expand-brackets "^2.1.4" - extend-shallow "^2.0.1" - fragment-cache "^0.2.1" - regex-not "^1.0.0" - snapdragon "^0.8.1" - to-regex "^3.0.1" - -extract-zip@^1.0.3: - version "1.7.0" - resolved "https://registry.yarnpkg.com/extract-zip/-/extract-zip-1.7.0.tgz#556cc3ae9df7f452c493a0cfb51cc30277940927" - integrity sha512-xoh5G1W/PB0/27lXgMQyIhP5DSY/LhoCsOyZgb+6iMmRtCwVBo55uKaMoEYrDCKQhWvqEip5ZPKAc6eFNyf/MA== - dependencies: - concat-stream "^1.6.2" - debug "^2.6.9" - mkdirp "^0.5.4" - yauzl "^2.10.0" - -extsprintf@1.3.0: - version "1.3.0" - resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.3.0.tgz#96918440e3041a7a414f8c52e3c574eb3c3e1e05" - integrity sha1-lpGEQOMEGnpBT4xS48V06zw+HgU= - -extsprintf@^1.2.0: - version "1.4.0" - resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.4.0.tgz#e2689f8f356fad62cca65a3a91c5df5f9551692f" - integrity sha1-4mifjzVvrWLMplo6kcXfX5VRaS8= - -fast-deep-equal@^3.1.1: - version "3.1.3" - resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" - integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q== - -fast-glob@^3.1.1, fast-glob@^3.2.4: - version "3.2.5" - resolved "https://registry.yarnpkg.com/fast-glob/-/fast-glob-3.2.5.tgz#7939af2a656de79a4f1901903ee8adcaa7cb9661" - integrity sha512-2DtFcgT68wiTTiwZ2hNdJfcHNke9XOfnwmBRWXhmeKM8rF0TGwmC/Qto3S7RoZKp5cilZbxzO5iTNTQsJ+EeDg== - dependencies: - "@nodelib/fs.stat" "^2.0.2" - "@nodelib/fs.walk" "^1.2.3" - glob-parent "^5.1.0" - merge2 "^1.3.0" - micromatch "^4.0.2" - picomatch "^2.2.1" - -fast-json-stable-stringify@2.1.0, fast-json-stable-stringify@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz#874bf69c6f404c2b5d99c481341399fd55892633" - integrity sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw== - -fastparse@^1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/fastparse/-/fastparse-1.1.2.tgz#91728c5a5942eced8531283c79441ee4122c35a9" - integrity sha512-483XLLxTVIwWK3QTrMGRqUfUpoOs/0hbQrl2oz4J0pAcm3A3bu84wxTFqGqkJzewCLdME38xJLJAxBABfQT8sQ== - -fastq@^1.6.0: - version "1.11.0" - resolved "https://registry.yarnpkg.com/fastq/-/fastq-1.11.0.tgz#bb9fb955a07130a918eb63c1f5161cc32a5d0858" - integrity sha512-7Eczs8gIPDrVzT+EksYBcupqMyxSHXXrHOLRRxU2/DicV8789MRBRR8+Hc2uWzUupOs4YS4JzBmBxjjCVBxD/g== - dependencies: - reusify "^1.0.4" - -faye-websocket@^0.10.0: - version "0.10.0" - resolved "https://registry.yarnpkg.com/faye-websocket/-/faye-websocket-0.10.0.tgz#4e492f8d04dfb6f89003507f6edbf2d501e7c6f4" - integrity sha1-TkkvjQTftviQA1B/btvy1QHnxvQ= - dependencies: - websocket-driver ">=0.5.1" - -faye-websocket@~0.11.1: - version "0.11.3" - resolved "https://registry.yarnpkg.com/faye-websocket/-/faye-websocket-0.11.3.tgz#5c0e9a8968e8912c286639fde977a8b209f2508e" - integrity sha512-D2y4bovYpzziGgbHYtGCMjlJM36vAl/y+xUyn1C+FVx8szd1E+86KwVw6XvYSzOP8iMpm1X0I4xJD+QtUb36OA== - dependencies: - websocket-driver ">=0.5.1" - -fd-slicer@~1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/fd-slicer/-/fd-slicer-1.1.0.tgz#25c7c89cb1f9077f8891bbe61d8f390eae256f1e" - integrity sha1-JcfInLH5B3+IkbvmHY85Dq4lbx4= - dependencies: - pend "~1.2.0" - -figgy-pudding@^3.4.1, figgy-pudding@^3.5.1: - version "3.5.2" - resolved "https://registry.yarnpkg.com/figgy-pudding/-/figgy-pudding-3.5.2.tgz#b4eee8148abb01dcf1d1ac34367d59e12fa61d6e" - integrity sha512-0btnI/H8f2pavGMN8w40mlSKOfTK2SVJmBfBeVIj3kNw0swwgzyRq0d5TJVOwodFmtvpPeWPN/MCcfuWF0Ezbw== - -figures@^3.0.0: - version "3.2.0" - resolved "https://registry.yarnpkg.com/figures/-/figures-3.2.0.tgz#625c18bd293c604dc4a8ddb2febf0c88341746af" - integrity sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg== - dependencies: - escape-string-regexp "^1.0.5" - -file-loader@6.1.1: - version "6.1.1" - resolved "https://registry.yarnpkg.com/file-loader/-/file-loader-6.1.1.tgz#a6f29dfb3f5933a1c350b2dbaa20ac5be0539baa" - integrity sha512-Klt8C4BjWSXYQAfhpYYkG4qHNTna4toMHEbWrI5IuVoxbU6uiDKeKAP99R8mmbJi3lvewn/jQBOgU4+NS3tDQw== - dependencies: - loader-utils "^2.0.0" - schema-utils "^3.0.0" - -file-uri-to-path@1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz#553a7b8446ff6f684359c445f1e37a05dacc33dd" - integrity sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw== - -fill-range@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-4.0.0.tgz#d544811d428f98eb06a63dc402d2403c328c38f7" - integrity sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc= - dependencies: - extend-shallow "^2.0.1" - is-number "^3.0.0" - repeat-string "^1.6.1" - to-regex-range "^2.1.0" - -fill-range@^7.0.1: - version "7.0.1" - resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.0.1.tgz#1919a6a7c75fe38b2c7c77e5198535da9acdda40" - integrity sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ== - dependencies: - to-regex-range "^5.0.1" - -finalhandler@1.1.2, finalhandler@~1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.1.2.tgz#b7e7d000ffd11938d0fdb053506f6ebabe9f587d" - integrity sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA== - dependencies: - debug "2.6.9" - encodeurl "~1.0.2" - escape-html "~1.0.3" - on-finished "~2.3.0" - parseurl "~1.3.3" - statuses "~1.5.0" - unpipe "~1.0.0" - -find-cache-dir@3.3.1, find-cache-dir@^3.3.1: - version "3.3.1" - resolved "https://registry.yarnpkg.com/find-cache-dir/-/find-cache-dir-3.3.1.tgz#89b33fad4a4670daa94f855f7fbe31d6d84fe880" - integrity sha512-t2GDMt3oGC/v+BMwzmllWDuJF/xcDtE5j/fCGbqDD7OLuJkj0cfh1YSA5VKPvwMeLFLNDBkwOKZ2X85jGLVftQ== - dependencies: - commondir "^1.0.1" - make-dir "^3.0.2" - pkg-dir "^4.1.0" - -find-cache-dir@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/find-cache-dir/-/find-cache-dir-2.1.0.tgz#8d0f94cd13fe43c6c7c261a0d86115ca918c05f7" - integrity sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ== - dependencies: - commondir "^1.0.1" - make-dir "^2.0.0" - pkg-dir "^3.0.0" - -find-up@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/find-up/-/find-up-3.0.0.tgz#49169f1d7993430646da61ecc5ae355c21c97b73" - integrity sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg== - dependencies: - locate-path "^3.0.0" - -find-up@^4.0.0, find-up@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/find-up/-/find-up-4.1.0.tgz#97afe7d6cdc0bc5928584b7c8d7b16e8a9aa5d19" - integrity sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw== - dependencies: - locate-path "^5.0.0" - path-exists "^4.0.0" - -flatted@^2.0.1, flatted@^2.0.2: - version "2.0.2" - resolved "https://registry.yarnpkg.com/flatted/-/flatted-2.0.2.tgz#4575b21e2bcee7434aa9be662f4b7b5f9c2b5138" - integrity sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA== - -flush-write-stream@^1.0.0: - version "1.1.1" - resolved "https://registry.yarnpkg.com/flush-write-stream/-/flush-write-stream-1.1.1.tgz#8dd7d873a1babc207d94ead0c2e0e44276ebf2e8" - integrity sha512-3Z4XhFZ3992uIq0XOqb9AreonueSYphE6oYbpt5+3u06JWklbsPkNv3ZKkP9Bz/r+1MWCaMoSQ28P85+1Yc77w== - dependencies: - inherits "^2.0.3" - readable-stream "^2.3.6" - -follow-redirects@^1.0.0: - version "1.13.3" - resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.13.3.tgz#e5598ad50174c1bc4e872301e82ac2cd97f90267" - integrity sha512-DUgl6+HDzB0iEptNQEXLx/KhTmDb8tZUHSeLqpnjpknR70H0nC2t9N73BK6fN4hOvJ84pKlIQVQ4k5FFlBedKA== - -for-in@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/for-in/-/for-in-1.0.2.tgz#81068d295a8142ec0ac726c6e2200c30fb6d5e80" - integrity sha1-gQaNKVqBQuwKxybG4iAMMPttXoA= - -forever-agent@~0.6.1: - version "0.6.1" - resolved "https://registry.yarnpkg.com/forever-agent/-/forever-agent-0.6.1.tgz#fbc71f0c41adeb37f96c577ad1ed42d8fdacca91" - integrity sha1-+8cfDEGt6zf5bFd60e1C2P2sypE= - -form-data@~2.3.2: - version "2.3.3" - resolved "https://registry.yarnpkg.com/form-data/-/form-data-2.3.3.tgz#dcce52c05f644f298c6a7ab936bd724ceffbf3a6" - integrity sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ== - dependencies: - asynckit "^0.4.0" - combined-stream "^1.0.6" - mime-types "^2.1.12" - -forwarded@~0.1.2: - version "0.1.2" - resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.1.2.tgz#98c23dab1175657b8c0573e8ceccd91b0ff18c84" - integrity sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ= - -fragment-cache@^0.2.1: - version "0.2.1" - resolved "https://registry.yarnpkg.com/fragment-cache/-/fragment-cache-0.2.1.tgz#4290fad27f13e89be7f33799c6bc5a0abfff0d19" - integrity sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk= - dependencies: - map-cache "^0.2.2" - -fresh@0.5.2: - version "0.5.2" - resolved "https://registry.yarnpkg.com/fresh/-/fresh-0.5.2.tgz#3d8cadd90d976569fa835ab1f8e4b23a105605a7" - integrity sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac= - -from2@^2.1.0: - version "2.3.0" - resolved "https://registry.yarnpkg.com/from2/-/from2-2.3.0.tgz#8bfb5502bde4a4d36cfdeea007fcca21d7e382af" - integrity sha1-i/tVAr3kpNNs/e6gB/zKIdfjgq8= - dependencies: - inherits "^2.0.1" - readable-stream "^2.0.0" - -fs-extra@4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-4.0.2.tgz#f91704c53d1b461f893452b0c307d9997647ab6b" - integrity sha1-+RcExT0bRh+JNFKwwwfZmXZHq2s= - dependencies: - graceful-fs "^4.1.2" - jsonfile "^4.0.0" - universalify "^0.1.0" - -fs-extra@^8.1.0: - version "8.1.0" - resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-8.1.0.tgz#49d43c45a88cd9677668cb7be1b46efdb8d2e1c0" - integrity sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g== - dependencies: - graceful-fs "^4.2.0" - jsonfile "^4.0.0" - universalify "^0.1.0" - -fs-minipass@^1.2.5: - version "1.2.7" - resolved "https://registry.yarnpkg.com/fs-minipass/-/fs-minipass-1.2.7.tgz#ccff8570841e7fe4265693da88936c55aed7f7c7" - integrity sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA== - dependencies: - minipass "^2.6.0" - -fs-minipass@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/fs-minipass/-/fs-minipass-2.1.0.tgz#7f5036fdbf12c63c169190cbe4199c852271f9fb" - integrity sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg== - dependencies: - minipass "^3.0.0" - -fs-write-stream-atomic@^1.0.8: - version "1.0.10" - resolved "https://registry.yarnpkg.com/fs-write-stream-atomic/-/fs-write-stream-atomic-1.0.10.tgz#b47df53493ef911df75731e70a9ded0189db40c9" - integrity sha1-tH31NJPvkR33VzHnCp3tAYnbQMk= - dependencies: - graceful-fs "^4.1.2" - iferr "^0.1.5" - imurmurhash "^0.1.4" - readable-stream "1 || 2" - -fs.realpath@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" - integrity sha1-FQStJSMVjKpA20onh8sBQRmU6k8= - -fsevents@^1.2.7: - version "1.2.13" - resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-1.2.13.tgz#f325cb0455592428bcf11b383370ef70e3bfcc38" - integrity sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw== - dependencies: - bindings "^1.5.0" - nan "^2.12.1" - -fsevents@~2.1.2: - version "2.1.3" - resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.1.3.tgz#fb738703ae8d2f9fe900c33836ddebee8b97f23e" - integrity sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ== - -fsevents@~2.3.1: - version "2.3.2" - resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.2.tgz#8a526f78b8fdf4623b709e0b975c52c24c02fd1a" - integrity sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA== - -function-bind@^1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.1.tgz#a56899d3ea3c9bab874bb9773b7c5ede92f4895d" - integrity sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A== - -genfun@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/genfun/-/genfun-5.0.0.tgz#9dd9710a06900a5c4a5bf57aca5da4e52fe76537" - integrity sha512-KGDOARWVga7+rnB3z9Sd2Letx515owfk0hSxHGuqjANb1M+x2bGZGqHLiozPsYMdM2OubeMni/Hpwmjq6qIUhA== - -gensync@^1.0.0-beta.1, gensync@^1.0.0-beta.2: - version "1.0.0-beta.2" - resolved "https://registry.yarnpkg.com/gensync/-/gensync-1.0.0-beta.2.tgz#32a6ee76c3d7f52d46b2b1ae5d93fea8580a25e0" - integrity sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg== - -get-caller-file@^2.0.1, get-caller-file@^2.0.5: - version "2.0.5" - resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz#4f94412a82db32f36e3b0b9741f8a97feb031f7e" - integrity sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg== - -get-intrinsic@^1.0.2, get-intrinsic@^1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.1.1.tgz#15f59f376f855c446963948f0d24cd3637b4abc6" - integrity sha512-kWZrnVM42QCiEA2Ig1bG8zjoIMOgxWwYCEeNdwY6Tv/cOSeGpcoX4pXHfKUxNKVoArnrEr2e9srnAxxGIraS9Q== - dependencies: - function-bind "^1.1.1" - has "^1.0.3" - has-symbols "^1.0.1" - -get-stream@^4.0.0, get-stream@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-4.1.0.tgz#c1b255575f3dc21d59bfc79cd3d2b46b1c3a54b5" - integrity sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w== - dependencies: - pump "^3.0.0" - -get-stream@^5.1.0: - version "5.2.0" - resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-5.2.0.tgz#4966a1795ee5ace65e706c4b7beb71257d6e22d3" - integrity sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA== - dependencies: - pump "^3.0.0" - -get-value@^2.0.3, get-value@^2.0.6: - version "2.0.6" - resolved "https://registry.yarnpkg.com/get-value/-/get-value-2.0.6.tgz#dc15ca1c672387ca76bd37ac0a395ba2042a2c28" - integrity sha1-3BXKHGcjh8p2vTesCjlbogQqLCg= - -getpass@^0.1.1: - version "0.1.7" - resolved "https://registry.yarnpkg.com/getpass/-/getpass-0.1.7.tgz#5eff8e3e684d569ae4cb2b1282604e8ba62149fa" - integrity sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo= - dependencies: - assert-plus "^1.0.0" - -glob-parent@^3.1.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-3.1.0.tgz#9e6af6299d8d3bd2bd40430832bd113df906c5ae" - integrity sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4= - dependencies: - is-glob "^3.1.0" - path-dirname "^1.0.0" - -glob-parent@^5.1.0, glob-parent@^5.1.1, glob-parent@~5.1.0: - version "5.1.2" - resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-5.1.2.tgz#869832c58034fe68a4093c17dc15e8340d8401c4" - integrity sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow== - dependencies: - is-glob "^4.0.1" - -glob@7.1.6, glob@^7.0.3, glob@^7.0.6, glob@^7.1.1, glob@^7.1.3, glob@^7.1.4, glob@^7.1.6: - version "7.1.6" - resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.6.tgz#141f33b81a7c2492e125594307480c46679278a6" - integrity sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA== - dependencies: - fs.realpath "^1.0.0" - inflight "^1.0.4" - inherits "2" - minimatch "^3.0.4" - once "^1.3.0" - path-is-absolute "^1.0.0" - -global-agent@^2.0.2: - version "2.1.12" - resolved "https://registry.yarnpkg.com/global-agent/-/global-agent-2.1.12.tgz#e4ae3812b731a9e81cbf825f9377ef450a8e4195" - integrity sha512-caAljRMS/qcDo69X9BfkgrihGUgGx44Fb4QQToNQjsiWh+YlQ66uqYVAdA8Olqit+5Ng0nkz09je3ZzANMZcjg== - dependencies: - boolean "^3.0.1" - core-js "^3.6.5" - es6-error "^4.1.1" - matcher "^3.0.0" - roarr "^2.15.3" - semver "^7.3.2" - serialize-error "^7.0.1" - -global-tunnel-ng@^2.7.1: - version "2.7.1" - resolved "https://registry.yarnpkg.com/global-tunnel-ng/-/global-tunnel-ng-2.7.1.tgz#d03b5102dfde3a69914f5ee7d86761ca35d57d8f" - integrity sha512-4s+DyciWBV0eK148wqXxcmVAbFVPqtc3sEtUE/GTQfuU80rySLcMhUmHKSHI7/LDj8q0gDYI1lIhRRB7ieRAqg== - dependencies: - encodeurl "^1.0.2" - lodash "^4.17.10" - npm-conf "^1.1.3" - tunnel "^0.0.6" - -globals@^11.1.0: - version "11.12.0" - resolved "https://registry.yarnpkg.com/globals/-/globals-11.12.0.tgz#ab8795338868a0babd8525758018c2a7eb95c42e" - integrity sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA== - -globalthis@^1.0.1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/globalthis/-/globalthis-1.0.2.tgz#2a235d34f4d8036219f7e34929b5de9e18166b8b" - integrity sha512-ZQnSFO1la8P7auIOQECnm0sSuoMeaSq0EEdXMBFF2QJO4uNcwbyhSgG3MruWNbFTqCLmxVwGOl7LZ9kASvHdeQ== - dependencies: - define-properties "^1.1.3" - -globby@^11.0.1: - version "11.0.2" - resolved "https://registry.yarnpkg.com/globby/-/globby-11.0.2.tgz#1af538b766a3b540ebfb58a32b2e2d5897321d83" - integrity sha512-2ZThXDvvV8fYFRVIxnrMQBipZQDr7MxKAmQK1vujaj9/7eF0efG7BPUKJ7jP7G5SLF37xKDXvO4S/KKLj/Z0og== - dependencies: - array-union "^2.1.0" - dir-glob "^3.0.1" - fast-glob "^3.1.1" - ignore "^5.1.4" - merge2 "^1.3.0" - slash "^3.0.0" - -globby@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/globby/-/globby-5.0.0.tgz#ebd84667ca0dbb330b99bcfc68eac2bc54370e0d" - integrity sha1-69hGZ8oNuzMLmbz8aOrCvFQ3Dg0= - dependencies: - array-union "^1.0.1" - arrify "^1.0.0" - glob "^7.0.3" - object-assign "^4.0.1" - pify "^2.0.0" - pinkie-promise "^2.0.0" - -globby@^6.1.0: - version "6.1.0" - resolved "https://registry.yarnpkg.com/globby/-/globby-6.1.0.tgz#f5a6d70e8395e21c858fb0489d64df02424d506c" - integrity sha1-9abXDoOV4hyFj7BInWTfAkJNUGw= - dependencies: - array-union "^1.0.1" - glob "^7.0.3" - object-assign "^4.0.1" - pify "^2.0.0" - pinkie-promise "^2.0.0" - -got@^9.6.0: - version "9.6.0" - resolved "https://registry.yarnpkg.com/got/-/got-9.6.0.tgz#edf45e7d67f99545705de1f7bbeeeb121765ed85" - integrity sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q== - dependencies: - "@sindresorhus/is" "^0.14.0" - "@szmarczak/http-timer" "^1.1.2" - cacheable-request "^6.0.0" - decompress-response "^3.3.0" - duplexer3 "^0.1.4" - get-stream "^4.1.0" - lowercase-keys "^1.0.1" - mimic-response "^1.0.1" - p-cancelable "^1.0.0" - to-readable-stream "^1.0.0" - url-parse-lax "^3.0.0" - -graceful-fs@^4.1.11, graceful-fs@^4.1.15, graceful-fs@^4.1.2, graceful-fs@^4.1.6, graceful-fs@^4.2.0, graceful-fs@^4.2.4: - version "4.2.6" - resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.6.tgz#ff040b2b0853b23c3d31027523706f1885d76bee" - integrity sha512-nTnJ528pbqxYanhpDYsi4Rd8MAeaBA67+RZ10CM1m3bTAVFEDcd5AuA4a6W5YkGZ1iNXHzZz8T6TBKLeBuNriQ== - -handle-thing@^2.0.0: - version "2.0.1" - resolved "https://registry.yarnpkg.com/handle-thing/-/handle-thing-2.0.1.tgz#857f79ce359580c340d43081cc648970d0bb234e" - integrity sha512-9Qn4yBxelxoh2Ow62nP+Ka/kMnOXRi8BXnRaUwezLNhqelnN49xKz4F/dPP8OYLxLxq6JDtZb2i9XznUQbNPTg== - -har-schema@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/har-schema/-/har-schema-2.0.0.tgz#a94c2224ebcac04782a0d9035521f24735b7ec92" - integrity sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI= - -har-validator@~5.1.3: - version "5.1.5" - resolved "https://registry.yarnpkg.com/har-validator/-/har-validator-5.1.5.tgz#1f0803b9f8cb20c0fa13822df1ecddb36bde1efd" - integrity sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w== - dependencies: - ajv "^6.12.3" - har-schema "^2.0.0" - -has-ansi@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/has-ansi/-/has-ansi-2.0.0.tgz#34f5049ce1ecdf2b0649af3ef24e45ed35416d91" - integrity sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE= - dependencies: - ansi-regex "^2.0.0" - -has-bigints@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/has-bigints/-/has-bigints-1.0.1.tgz#64fe6acb020673e3b78db035a5af69aa9d07b113" - integrity sha512-LSBS2LjbNBTf6287JEbEzvJgftkF5qFkmCo9hDRpAzKhUOlJ+hx8dd4USs00SgsUNwc4617J9ki5YtEClM2ffA== - -has-binary2@~1.0.2: - version "1.0.3" - resolved "https://registry.yarnpkg.com/has-binary2/-/has-binary2-1.0.3.tgz#7776ac627f3ea77250cfc332dab7ddf5e4f5d11d" - integrity sha512-G1LWKhDSvhGeAQ8mPVQlqNcOB2sJdwATtZKl2pDKKHfpf/rYj24lkinxf69blJbnsvtqqNU+L3SL50vzZhXOnw== - dependencies: - isarray "2.0.1" - -has-cors@1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/has-cors/-/has-cors-1.1.0.tgz#5e474793f7ea9843d1bb99c23eef49ff126fff39" - integrity sha1-XkdHk/fqmEPRu5nCPu9J/xJv/zk= - -has-flag@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd" - integrity sha1-tdRU3CGZriJWmfNGfloH87lVuv0= - -has-flag@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b" - integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ== - -has-symbols@^1.0.0, has-symbols@^1.0.1, has-symbols@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.2.tgz#165d3070c00309752a1236a479331e3ac56f1423" - integrity sha512-chXa79rL/UC2KlX17jo3vRGz0azaWEx5tGqZg5pO3NUyEJVB17dMruQlzCCOfUvElghKcm5194+BCRvi2Rv/Gw== - -has-value@^0.3.1: - version "0.3.1" - resolved "https://registry.yarnpkg.com/has-value/-/has-value-0.3.1.tgz#7b1f58bada62ca827ec0a2078025654845995e1f" - integrity sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8= - dependencies: - get-value "^2.0.3" - has-values "^0.1.4" - isobject "^2.0.0" - -has-value@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/has-value/-/has-value-1.0.0.tgz#18b281da585b1c5c51def24c930ed29a0be6b177" - integrity sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc= - dependencies: - get-value "^2.0.6" - has-values "^1.0.0" - isobject "^3.0.0" - -has-values@^0.1.4: - version "0.1.4" - resolved "https://registry.yarnpkg.com/has-values/-/has-values-0.1.4.tgz#6d61de95d91dfca9b9a02089ad384bff8f62b771" - integrity sha1-bWHeldkd/Km5oCCJrThL/49it3E= - -has-values@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/has-values/-/has-values-1.0.0.tgz#95b0b63fec2146619a6fe57fe75628d5a39efe4f" - integrity sha1-lbC2P+whRmGab+V/51Yo1aOe/k8= - dependencies: - is-number "^3.0.0" - kind-of "^4.0.0" - -has@^1.0.0, has@^1.0.3: - version "1.0.3" - resolved "https://registry.yarnpkg.com/has/-/has-1.0.3.tgz#722d7cbfc1f6aa8241f16dd814e011e1f41e8796" - integrity sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw== - dependencies: - function-bind "^1.1.1" - -hash-base@^3.0.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/hash-base/-/hash-base-3.1.0.tgz#55c381d9e06e1d2997a883b4a3fddfe7f0d3af33" - integrity sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA== - dependencies: - inherits "^2.0.4" - readable-stream "^3.6.0" - safe-buffer "^5.2.0" - -hash.js@^1.0.0, hash.js@^1.0.3: - version "1.1.7" - resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.7.tgz#0babca538e8d4ee4a0f8988d68866537a003cf42" - integrity sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA== - dependencies: - inherits "^2.0.3" - minimalistic-assert "^1.0.1" - -hex-color-regex@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/hex-color-regex/-/hex-color-regex-1.1.0.tgz#4c06fccb4602fe2602b3c93df82d7e7dbf1a8a8e" - integrity sha512-l9sfDFsuqtOqKDsQdqrMRk0U85RZc0RtOR9yPI7mRVOa4FsR/BVnZ0shmQRM96Ji99kYZP/7hn1cedc1+ApsTQ== - -hmac-drbg@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/hmac-drbg/-/hmac-drbg-1.0.1.tgz#d2745701025a6c775a6c545793ed502fc0c649a1" - integrity sha1-0nRXAQJabHdabFRXk+1QL8DGSaE= - dependencies: - hash.js "^1.0.3" - minimalistic-assert "^1.0.0" - minimalistic-crypto-utils "^1.0.1" - -hosted-git-info@^2.1.4, hosted-git-info@^2.7.1: - version "2.8.8" - resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-2.8.8.tgz#7539bd4bc1e0e0a895815a2e0262420b12858488" - integrity sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg== - -hosted-git-info@^3.0.6: - version "3.0.8" - resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-3.0.8.tgz#6e35d4cc87af2c5f816e4cb9ce350ba87a3f370d" - integrity sha512-aXpmwoOhRBrw6X3j0h5RloK4x1OzsxMPyxqIHyNfSe2pypkVTZFpEiRoSipPEPlMrh0HW/XsjkJ5WgnCirpNUw== - dependencies: - lru-cache "^6.0.0" - -hpack.js@^2.1.6: - version "2.1.6" - resolved "https://registry.yarnpkg.com/hpack.js/-/hpack.js-2.1.6.tgz#87774c0949e513f42e84575b3c45681fade2a0b2" - integrity sha1-h3dMCUnlE/QuhFdbPEVoH63ioLI= - dependencies: - inherits "^2.0.1" - obuf "^1.0.0" - readable-stream "^2.0.1" - wbuf "^1.1.0" - -hsl-regex@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/hsl-regex/-/hsl-regex-1.0.0.tgz#d49330c789ed819e276a4c0d272dffa30b18fe6e" - integrity sha1-1JMwx4ntgZ4nakwNJy3/owsY/m4= - -hsla-regex@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/hsla-regex/-/hsla-regex-1.0.0.tgz#c1ce7a3168c8c6614033a4b5f7877f3b225f9c38" - integrity sha1-wc56MWjIxmFAM6S194d/OyJfnDg= - -html-comment-regex@^1.1.0: - version "1.1.2" - resolved "https://registry.yarnpkg.com/html-comment-regex/-/html-comment-regex-1.1.2.tgz#97d4688aeb5c81886a364faa0cad1dda14d433a7" - integrity sha512-P+M65QY2JQ5Y0G9KKdlDpo0zK+/OHptU5AaBwUfAIDJZk1MYf32Frm84EcOytfJE0t5JvkAnKlmjsXDnWzCJmQ== - -html-entities@^1.3.1: - version "1.4.0" - resolved "https://registry.yarnpkg.com/html-entities/-/html-entities-1.4.0.tgz#cfbd1b01d2afaf9adca1b10ae7dffab98c71d2dc" - integrity sha512-8nxjcBcd8wovbeKx7h3wTji4e6+rhaVuPNpMqwWgnHh+N9ToqsCs6XztWRBPQ+UtzsoMAdKZtUENoVzU/EMtZA== - -html-escaper@^2.0.0: - version "2.0.2" - resolved "https://registry.yarnpkg.com/html-escaper/-/html-escaper-2.0.2.tgz#dfd60027da36a36dfcbe236262c00a5822681453" - integrity sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg== - -http-cache-semantics@^3.8.1: - version "3.8.1" - resolved "https://registry.yarnpkg.com/http-cache-semantics/-/http-cache-semantics-3.8.1.tgz#39b0e16add9b605bf0a9ef3d9daaf4843b4cacd2" - integrity sha512-5ai2iksyV8ZXmnZhHH4rWPoxxistEexSi5936zIQ1bnNTW5VnA85B6P/VpXiRM017IgRvb2kKo1a//y+0wSp3w== - -http-cache-semantics@^4.0.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz#49e91c5cbf36c9b94bcfcd71c23d5249ec74e390" - integrity sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ== - -http-deceiver@^1.2.7: - version "1.2.7" - resolved "https://registry.yarnpkg.com/http-deceiver/-/http-deceiver-1.2.7.tgz#fa7168944ab9a519d337cb0bec7284dc3e723d87" - integrity sha1-+nFolEq5pRnTN8sL7HKE3D5yPYc= - -http-errors@1.7.2: - version "1.7.2" - resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.7.2.tgz#4f5029cf13239f31036e5b2e55292bcfbcc85c8f" - integrity sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg== - dependencies: - depd "~1.1.2" - inherits "2.0.3" - setprototypeof "1.1.1" - statuses ">= 1.5.0 < 2" - toidentifier "1.0.0" - -http-errors@~1.6.2: - version "1.6.3" - resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.6.3.tgz#8b55680bb4be283a0b5bf4ea2e38580be1d9320d" - integrity sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0= - dependencies: - depd "~1.1.2" - inherits "2.0.3" - setprototypeof "1.1.0" - statuses ">= 1.4.0 < 2" - -http-errors@~1.7.2: - version "1.7.3" - resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.7.3.tgz#6c619e4f9c60308c38519498c14fbb10aacebb06" - integrity sha512-ZTTX0MWrsQ2ZAhA1cejAwDLycFsd7I7nVtnkT3Ol0aqodaKW+0CTZDQ1uBv5whptCnc8e8HeRRJxRs0kmm/Qfw== - dependencies: - depd "~1.1.2" - inherits "2.0.4" - setprototypeof "1.1.1" - statuses ">= 1.5.0 < 2" - toidentifier "1.0.0" - -http-parser-js@>=0.5.1: - version "0.5.3" - resolved "https://registry.yarnpkg.com/http-parser-js/-/http-parser-js-0.5.3.tgz#01d2709c79d41698bb01d4decc5e9da4e4a033d9" - integrity sha512-t7hjvef/5HEK7RWTdUzVUhl8zkEu+LlaE0IYzdMuvbSDipxBRpOn4Uhw8ZyECEa808iVT8XCjzo6xmYt4CiLZg== - -http-proxy-agent@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/http-proxy-agent/-/http-proxy-agent-2.1.0.tgz#e4821beef5b2142a2026bd73926fe537631c5405" - integrity sha512-qwHbBLV7WviBl0rQsOzH6o5lwyOIvwp/BdFnvVxXORldu5TmjFfjzBcWUWS5kWAZhmv+JtiDhSuQCp4sBfbIgg== - dependencies: - agent-base "4" - debug "3.1.0" - -http-proxy-middleware@0.19.1: - version "0.19.1" - resolved "https://registry.yarnpkg.com/http-proxy-middleware/-/http-proxy-middleware-0.19.1.tgz#183c7dc4aa1479150306498c210cdaf96080a43a" - integrity sha512-yHYTgWMQO8VvwNS22eLLloAkvungsKdKTLO8AJlftYIKNfJr3GK3zK0ZCfzDDGUBttdGc8xFy1mCitvNKQtC3Q== - dependencies: - http-proxy "^1.17.0" - is-glob "^4.0.0" - lodash "^4.17.11" - micromatch "^3.1.10" - -http-proxy@^1.17.0, http-proxy@^1.18.1: - version "1.18.1" - resolved "https://registry.yarnpkg.com/http-proxy/-/http-proxy-1.18.1.tgz#401541f0534884bbf95260334e72f88ee3976549" - integrity sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ== - dependencies: - eventemitter3 "^4.0.0" - follow-redirects "^1.0.0" - requires-port "^1.0.0" - -http-signature@~1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/http-signature/-/http-signature-1.2.0.tgz#9aecd925114772f3d95b65a60abb8f7c18fbace1" - integrity sha1-muzZJRFHcvPZW2WmCruPfBj7rOE= - dependencies: - assert-plus "^1.0.0" - jsprim "^1.2.2" - sshpk "^1.7.0" - -https-browserify@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/https-browserify/-/https-browserify-1.0.0.tgz#ec06c10e0a34c0f2faf199f7fd7fc78fffd03c73" - integrity sha1-7AbBDgo0wPL68Zn3/X/Hj//QPHM= - -https-proxy-agent@^2.2.1, https-proxy-agent@^2.2.3: - version "2.2.4" - resolved "https://registry.yarnpkg.com/https-proxy-agent/-/https-proxy-agent-2.2.4.tgz#4ee7a737abd92678a293d9b34a1af4d0d08c787b" - integrity sha512-OmvfoQ53WLjtA9HeYP9RNrWMJzzAz1JGaSFr1nijg0PVR1JaD/xbJq1mdEIIlxGpXp9eSe/O2LgU9DJmTPd0Eg== - dependencies: - agent-base "^4.3.0" - debug "^3.1.0" - -humanize-ms@^1.2.1: - version "1.2.1" - resolved "https://registry.yarnpkg.com/humanize-ms/-/humanize-ms-1.2.1.tgz#c46e3159a293f6b896da29316d8b6fe8bb79bbed" - integrity sha1-xG4xWaKT9riW2ikxbYtv6Lt5u+0= - dependencies: - ms "^2.0.0" - -iconv-lite@0.4.24, iconv-lite@^0.4.24: - version "0.4.24" - resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b" - integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA== - dependencies: - safer-buffer ">= 2.1.2 < 3" - -iconv-lite@^0.6.2: - version "0.6.2" - resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.6.2.tgz#ce13d1875b0c3a674bd6a04b7f76b01b1b6ded01" - integrity sha512-2y91h5OpQlolefMPmUlivelittSWy0rP+oYVpn6A7GwVHNE8AWzoYOBNmlwks3LobaJxgHCYZAnyNo2GgpNRNQ== - dependencies: - safer-buffer ">= 2.1.2 < 3.0.0" - -icss-utils@^4.0.0, icss-utils@^4.1.1: - version "4.1.1" - resolved "https://registry.yarnpkg.com/icss-utils/-/icss-utils-4.1.1.tgz#21170b53789ee27447c2f47dd683081403f9a467" - integrity sha512-4aFq7wvWyMHKgxsH8QQtGpvbASCf+eM3wPRLI6R+MgAnTCZ6STYsRvttLvRWK0Nfif5piF394St3HeJDaljGPA== - dependencies: - postcss "^7.0.14" - -ieee754@^1.1.4: - version "1.2.1" - resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352" - integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA== - -iferr@^0.1.5: - version "0.1.5" - resolved "https://registry.yarnpkg.com/iferr/-/iferr-0.1.5.tgz#c60eed69e6d8fdb6b3104a1fcbca1c192dc5b501" - integrity sha1-xg7taebY/bazEEofy8ocGS3FtQE= - -ignore-walk@^3.0.1: - version "3.0.3" - resolved "https://registry.yarnpkg.com/ignore-walk/-/ignore-walk-3.0.3.tgz#017e2447184bfeade7c238e4aefdd1e8f95b1e37" - integrity sha512-m7o6xuOaT1aqheYHKf8W6J5pYH85ZI9w077erOzLje3JsB1gkafkAhHHY19dqjulgIZHFm32Cp5uNZgcQqdJKw== - dependencies: - minimatch "^3.0.4" - -ignore@^5.1.4: - version "5.1.8" - resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.1.8.tgz#f150a8b50a34289b33e22f5889abd4d8016f0e57" - integrity sha512-BMpfD7PpiETpBl/A6S498BaIJ6Y/ABT93ETbby2fP00v4EbvPBXWEoaR1UBPKs3iR53pJY7EtZk5KACI57i1Uw== - -image-size@~0.5.0: - version "0.5.5" - resolved "https://registry.yarnpkg.com/image-size/-/image-size-0.5.5.tgz#09dfd4ab9d20e29eb1c3e80b8990378df9e3cb9c" - integrity sha1-Cd/Uq50g4p6xw+gLiZA3jfnjy5w= - -immediate@~3.0.5: - version "3.0.6" - resolved "https://registry.yarnpkg.com/immediate/-/immediate-3.0.6.tgz#9db1dbd0faf8de6fbe0f5dd5e56bb606280de69b" - integrity sha1-nbHb0Pr43m++D13V5Wu2BigN5ps= - -import-fresh@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/import-fresh/-/import-fresh-2.0.0.tgz#d81355c15612d386c61f9ddd3922d4304822a546" - integrity sha1-2BNVwVYS04bGH53dOSLUMEgipUY= - dependencies: - caller-path "^2.0.0" - resolve-from "^3.0.0" - -import-fresh@^3.2.1: - version "3.3.0" - resolved "https://registry.yarnpkg.com/import-fresh/-/import-fresh-3.3.0.tgz#37162c25fcb9ebaa2e6e53d5b4d88ce17d9e0c2b" - integrity sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw== - dependencies: - parent-module "^1.0.0" - resolve-from "^4.0.0" - -import-local@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/import-local/-/import-local-2.0.0.tgz#55070be38a5993cf18ef6db7e961f5bee5c5a09d" - integrity sha512-b6s04m3O+s3CGSbqDIyP4R6aAwAeYlVq9+WUWep6iHa8ETRf9yei1U48C5MmfJmV9AiLYYBKPMq/W+/WRpQmCQ== - dependencies: - pkg-dir "^3.0.0" - resolve-cwd "^2.0.0" - -imurmurhash@^0.1.4: - version "0.1.4" - resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz#9218b9b2b928a238b13dc4fb6b6d576f231453ea" - integrity sha1-khi5srkoojixPcT7a21XbyMUU+o= - -indent-string@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/indent-string/-/indent-string-4.0.0.tgz#624f8f4497d619b2d9768531d58f4122854d7251" - integrity sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg== - -indexes-of@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/indexes-of/-/indexes-of-1.0.1.tgz#f30f716c8e2bd346c7b67d3df3915566a7c05607" - integrity sha1-8w9xbI4r00bHtn0985FVZqfAVgc= - -indexof@0.0.1: - version "0.0.1" - resolved "https://registry.yarnpkg.com/indexof/-/indexof-0.0.1.tgz#82dc336d232b9062179d05ab3293a66059fd435d" - integrity sha1-gtwzbSMrkGIXnQWrMpOmYFn9Q10= - -infer-owner@^1.0.3, infer-owner@^1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/infer-owner/-/infer-owner-1.0.4.tgz#c4cefcaa8e51051c2a40ba2ce8a3d27295af9467" - integrity sha512-IClj+Xz94+d7irH5qRyfJonOdfTzuDaifE6ZPWfx0N0+/ATZCbuTPq2prFl526urkQd90WyUKIh1DfBQ2hMz9A== - -inflight@^1.0.4: - version "1.0.6" - resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" - integrity sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk= - dependencies: - once "^1.3.0" - wrappy "1" - -inherits@2, inherits@2.0.4, inherits@^2.0.1, inherits@^2.0.3, inherits@^2.0.4, inherits@~2.0.1, inherits@~2.0.3: - version "2.0.4" - resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" - integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== - -inherits@2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.1.tgz#b17d08d326b4423e568eff719f91b0b1cbdf69f1" - integrity sha1-sX0I0ya0Qj5Wjv9xn5GwscvfafE= - -inherits@2.0.3: - version "2.0.3" - resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.3.tgz#633c2c83e3da42a502f52466022480f4208261de" - integrity sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4= - -ini@1.3.6: - version "1.3.6" - resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.6.tgz#f1c46a2a93a253e7b3905115e74d527cd23061a1" - integrity sha512-IZUoxEjNjubzrmvzZU4lKP7OnYmX72XRl3sqkfJhBKweKi5rnGi5+IUdlj/H1M+Ip5JQ1WzaDMOBRY90Ajc5jg== - -ini@^1.3.4: - version "1.3.8" - resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.8.tgz#a29da425b48806f34767a4efce397269af28432c" - integrity sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew== - -inquirer@7.3.3: - version "7.3.3" - resolved "https://registry.yarnpkg.com/inquirer/-/inquirer-7.3.3.tgz#04d176b2af04afc157a83fd7c100e98ee0aad003" - integrity sha512-JG3eIAj5V9CwcGvuOmoo6LB9kbAYT8HXffUl6memuszlwDC/qvFAJw49XJ5NROSFNPxp3iQg1GqkFhaY/CR0IA== - dependencies: - ansi-escapes "^4.2.1" - chalk "^4.1.0" - cli-cursor "^3.1.0" - cli-width "^3.0.0" - external-editor "^3.0.3" - figures "^3.0.0" - lodash "^4.17.19" - mute-stream "0.0.8" - run-async "^2.4.0" - rxjs "^6.6.0" - string-width "^4.1.0" - strip-ansi "^6.0.0" - through "^2.3.6" - -internal-ip@^4.3.0: - version "4.3.0" - resolved "https://registry.yarnpkg.com/internal-ip/-/internal-ip-4.3.0.tgz#845452baad9d2ca3b69c635a137acb9a0dad0907" - integrity sha512-S1zBo1D6zcsyuC6PMmY5+55YMILQ9av8lotMx447Bq6SAgo/sDK6y6uUKmuYhW7eacnIhFfsPmCNYdDzsnnDCg== - dependencies: - default-gateway "^4.2.0" - ipaddr.js "^1.9.0" - -ip-regex@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/ip-regex/-/ip-regex-2.1.0.tgz#fa78bf5d2e6913c911ce9f819ee5146bb6d844e9" - integrity sha1-+ni/XS5pE8kRzp+BnuUUa7bYROk= - -ip@1.1.5, ip@^1.1.0, ip@^1.1.5: - version "1.1.5" - resolved "https://registry.yarnpkg.com/ip/-/ip-1.1.5.tgz#bdded70114290828c0a039e72ef25f5aaec4354a" - integrity sha1-vd7XARQpCCjAoDnnLvJfWq7ENUo= - -ipaddr.js@1.9.1, ipaddr.js@^1.9.0: - version "1.9.1" - resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-1.9.1.tgz#bff38543eeb8984825079ff3a2a8e6cbd46781b3" - integrity sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g== - -is-absolute-url@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/is-absolute-url/-/is-absolute-url-2.1.0.tgz#50530dfb84fcc9aa7dbe7852e83a37b93b9f2aa6" - integrity sha1-UFMN+4T8yap9vnhS6Do3uTufKqY= - -is-absolute-url@^3.0.3: - version "3.0.3" - resolved "https://registry.yarnpkg.com/is-absolute-url/-/is-absolute-url-3.0.3.tgz#96c6a22b6a23929b11ea0afb1836c36ad4a5d698" - integrity sha512-opmNIX7uFnS96NtPmhWQgQx6/NYFgsUXYMllcfzwWKUMwfo8kku1TvE6hkNcH+Q1ts5cMVrsY7j0bxXQDciu9Q== - -is-accessor-descriptor@^0.1.6: - version "0.1.6" - resolved "https://registry.yarnpkg.com/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz#a9e12cb3ae8d876727eeef3843f8a0897b5c98d6" - integrity sha1-qeEss66Nh2cn7u84Q/igiXtcmNY= - dependencies: - kind-of "^3.0.2" - -is-accessor-descriptor@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz#169c2f6d3df1f992618072365c9b0ea1f6878656" - integrity sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ== - dependencies: - kind-of "^6.0.0" - -is-arguments@^1.0.4: - version "1.1.0" - resolved "https://registry.yarnpkg.com/is-arguments/-/is-arguments-1.1.0.tgz#62353031dfbee07ceb34656a6bde59efecae8dd9" - integrity sha512-1Ij4lOMPl/xB5kBDn7I+b2ttPMKa8szhEIrXDuXQD/oe3HJLTLhqhgGspwgyGd6MOywBUqVvYicF72lkgDnIHg== - dependencies: - call-bind "^1.0.0" - -is-arrayish@^0.2.1: - version "0.2.1" - resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.2.1.tgz#77c99840527aa8ecb1a8ba697b80645a7a926a9d" - integrity sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0= - -is-arrayish@^0.3.1: - version "0.3.2" - resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.3.2.tgz#4574a2ae56f7ab206896fb431eaeed066fdf8f03" - integrity sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ== - -is-bigint@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/is-bigint/-/is-bigint-1.0.1.tgz#6923051dfcbc764278540b9ce0e6b3213aa5ebc2" - integrity sha512-J0ELF4yHFxHy0cmSxZuheDOz2luOdVvqjwmEcj8H/L1JHeuEDSDbeRP+Dk9kFVk5RTFzbucJ2Kb9F7ixY2QaCg== - -is-binary-path@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/is-binary-path/-/is-binary-path-1.0.1.tgz#75f16642b480f187a711c814161fd3a4a7655898" - integrity sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg= - dependencies: - binary-extensions "^1.0.0" - -is-binary-path@~2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/is-binary-path/-/is-binary-path-2.1.0.tgz#ea1f7f3b80f064236e83470f86c09c254fb45b09" - integrity sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw== - dependencies: - binary-extensions "^2.0.0" - -is-boolean-object@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/is-boolean-object/-/is-boolean-object-1.1.0.tgz#e2aaad3a3a8fca34c28f6eee135b156ed2587ff0" - integrity sha512-a7Uprx8UtD+HWdyYwnD1+ExtTgqQtD2k/1yJgtXP6wnMm8byhkoTZRl+95LLThpzNZJ5aEvi46cdH+ayMFRwmA== - dependencies: - call-bind "^1.0.0" - -is-buffer@^1.1.5: - version "1.1.6" - resolved "https://registry.yarnpkg.com/is-buffer/-/is-buffer-1.1.6.tgz#efaa2ea9daa0d7ab2ea13a97b2b8ad51fefbe8be" - integrity sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w== - -is-callable@^1.1.4, is-callable@^1.2.3: - version "1.2.3" - resolved "https://registry.yarnpkg.com/is-callable/-/is-callable-1.2.3.tgz#8b1e0500b73a1d76c70487636f368e519de8db8e" - integrity sha512-J1DcMe8UYTBSrKezuIUTUwjXsho29693unXM2YhJUTR2txK/eG47bvNa/wipPFmZFgr/N6f1GA66dv0mEyTIyQ== - -is-color-stop@^1.0.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/is-color-stop/-/is-color-stop-1.1.0.tgz#cfff471aee4dd5c9e158598fbe12967b5cdad345" - integrity sha1-z/9HGu5N1cnhWFmPvhKWe1za00U= - dependencies: - css-color-names "^0.0.4" - hex-color-regex "^1.1.0" - hsl-regex "^1.0.0" - hsla-regex "^1.0.0" - rgb-regex "^1.0.1" - rgba-regex "^1.0.0" - -is-core-module@^2.0.0, is-core-module@^2.2.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.2.0.tgz#97037ef3d52224d85163f5597b2b63d9afed981a" - integrity sha512-XRAfAdyyY5F5cOXn7hYQDqh2Xmii+DEfIcQGxK/uNwMHhIkPWO0g8msXcbzLe+MpGoR951MlqM/2iIlU4vKDdQ== - dependencies: - has "^1.0.3" - -is-data-descriptor@^0.1.4: - version "0.1.4" - resolved "https://registry.yarnpkg.com/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz#0b5ee648388e2c860282e793f1856fec3f301b56" - integrity sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y= - dependencies: - kind-of "^3.0.2" - -is-data-descriptor@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz#d84876321d0e7add03990406abbbbd36ba9268c7" - integrity sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ== - dependencies: - kind-of "^6.0.0" - -is-date-object@^1.0.1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/is-date-object/-/is-date-object-1.0.2.tgz#bda736f2cd8fd06d32844e7743bfa7494c3bfd7e" - integrity sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g== - -is-descriptor@^0.1.0: - version "0.1.6" - resolved "https://registry.yarnpkg.com/is-descriptor/-/is-descriptor-0.1.6.tgz#366d8240dde487ca51823b1ab9f07a10a78251ca" - integrity sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg== - dependencies: - is-accessor-descriptor "^0.1.6" - is-data-descriptor "^0.1.4" - kind-of "^5.0.0" - -is-descriptor@^1.0.0, is-descriptor@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/is-descriptor/-/is-descriptor-1.0.2.tgz#3b159746a66604b04f8c81524ba365c5f14d86ec" - integrity sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg== - dependencies: - is-accessor-descriptor "^1.0.0" - is-data-descriptor "^1.0.0" - kind-of "^6.0.2" - -is-directory@^0.3.1: - version "0.3.1" - resolved "https://registry.yarnpkg.com/is-directory/-/is-directory-0.3.1.tgz#61339b6f2475fc772fd9c9d83f5c8575dc154ae1" - integrity sha1-YTObbyR1/Hcv2cnYP1yFddwVSuE= - -is-docker@^2.0.0: - version "2.1.1" - resolved "https://registry.yarnpkg.com/is-docker/-/is-docker-2.1.1.tgz#4125a88e44e450d384e09047ede71adc2d144156" - integrity sha512-ZOoqiXfEwtGknTiuDEy8pN2CfE3TxMHprvNer1mXiqwkOT77Rw3YVrUQ52EqAOU3QAWDQ+bQdx7HJzrv7LS2Hw== - -is-extendable@^0.1.0, is-extendable@^0.1.1: - version "0.1.1" - resolved "https://registry.yarnpkg.com/is-extendable/-/is-extendable-0.1.1.tgz#62b110e289a471418e3ec36a617d472e301dfc89" - integrity sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik= - -is-extendable@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/is-extendable/-/is-extendable-1.0.1.tgz#a7470f9e426733d81bd81e1155264e3a3507cab4" - integrity sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA== - dependencies: - is-plain-object "^2.0.4" - -is-extglob@^2.1.0, is-extglob@^2.1.1: - version "2.1.1" - resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2" - integrity sha1-qIwCU1eR8C7TfHahueqXc8gz+MI= - -is-fullwidth-code-point@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz#a3b30a5c4f199183167aaab93beefae3ddfb654f" - integrity sha1-o7MKXE8ZkYMWeqq5O+764937ZU8= - -is-fullwidth-code-point@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d" - integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg== - -is-glob@^3.1.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-3.1.0.tgz#7ba5ae24217804ac70707b96922567486cc3e84a" - integrity sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo= - dependencies: - is-extglob "^2.1.0" - -is-glob@^4.0.0, is-glob@^4.0.1, is-glob@~4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.1.tgz#7567dbe9f2f5e2467bc77ab83c4a29482407a5dc" - integrity sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg== - dependencies: - is-extglob "^2.1.1" - -is-interactive@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/is-interactive/-/is-interactive-1.0.0.tgz#cea6e6ae5c870a7b0a0004070b7b587e0252912e" - integrity sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w== - -is-negative-zero@^2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/is-negative-zero/-/is-negative-zero-2.0.1.tgz#3de746c18dda2319241a53675908d8f766f11c24" - integrity sha512-2z6JzQvZRa9A2Y7xC6dQQm4FSTSTNWjKIYYTt4246eMTJmIo0Q+ZyOsU66X8lxK1AbB92dFeglPLrhwpeRKO6w== - -is-number-object@^1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/is-number-object/-/is-number-object-1.0.4.tgz#36ac95e741cf18b283fc1ddf5e83da798e3ec197" - integrity sha512-zohwelOAur+5uXtk8O3GPQ1eAcu4ZX3UwxQhUlfFFMNpUd83gXgjbhJh6HmB6LUNV/ieOLQuDwJO3dWJosUeMw== - -is-number@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/is-number/-/is-number-3.0.0.tgz#24fd6201a4782cf50561c810276afc7d12d71195" - integrity sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU= - dependencies: - kind-of "^3.0.2" - -is-number@^7.0.0: - version "7.0.0" - resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b" - integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng== - -is-obj@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-2.0.0.tgz#473fb05d973705e3fd9620545018ca8e22ef4982" - integrity sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w== - -is-path-cwd@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/is-path-cwd/-/is-path-cwd-1.0.0.tgz#d225ec23132e89edd38fda767472e62e65f1106d" - integrity sha1-0iXsIxMuie3Tj9p2dHLmLmXxEG0= - -is-path-cwd@^2.0.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/is-path-cwd/-/is-path-cwd-2.2.0.tgz#67d43b82664a7b5191fd9119127eb300048a9fdb" - integrity sha512-w942bTcih8fdJPJmQHFzkS76NEP8Kzzvmw92cXsazb8intwLqPibPPdXf4ANdKV3rYMuuQYGIWtvz9JilB3NFQ== - -is-path-in-cwd@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/is-path-in-cwd/-/is-path-in-cwd-1.0.1.tgz#5ac48b345ef675339bd6c7a48a912110b241cf52" - integrity sha512-FjV1RTW48E7CWM7eE/J2NJvAEEVektecDBVBE5Hh3nM1Jd0kvhHtX68Pr3xsDf857xt3Y4AkwVULK1Vku62aaQ== - dependencies: - is-path-inside "^1.0.0" - -is-path-in-cwd@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/is-path-in-cwd/-/is-path-in-cwd-2.1.0.tgz#bfe2dca26c69f397265a4009963602935a053acb" - integrity sha512-rNocXHgipO+rvnP6dk3zI20RpOtrAM/kzbB258Uw5BWr3TpXi861yzjo16Dn4hUox07iw5AyeMLHWsujkjzvRQ== - dependencies: - is-path-inside "^2.1.0" - -is-path-inside@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-1.0.1.tgz#8ef5b7de50437a3fdca6b4e865ef7aa55cb48036" - integrity sha1-jvW33lBDej/cprToZe96pVy0gDY= - dependencies: - path-is-inside "^1.0.1" - -is-path-inside@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-2.1.0.tgz#7c9810587d659a40d27bcdb4d5616eab059494b2" - integrity sha512-wiyhTzfDWsvwAW53OBWF5zuvaOGlZ6PwYxAbPVDhpm+gM09xKQGjBq/8uYN12aDvMxnAnq3dxTyoSoRNmg5YFg== - dependencies: - path-is-inside "^1.0.2" - -is-plain-object@^2.0.3, is-plain-object@^2.0.4: - version "2.0.4" - resolved "https://registry.yarnpkg.com/is-plain-object/-/is-plain-object-2.0.4.tgz#2c163b3fafb1b606d9d17928f05c2a1c38e07677" - integrity sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og== - dependencies: - isobject "^3.0.1" - -is-regex@^1.0.4, is-regex@^1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/is-regex/-/is-regex-1.1.2.tgz#81c8ebde4db142f2cf1c53fc86d6a45788266251" - integrity sha512-axvdhb5pdhEVThqJzYXwMlVuZwC+FF2DpcOhTS+y/8jVq4trxyPgfcwIxIKiyeuLlSQYKkmUaPQJ8ZE4yNKXDg== - dependencies: - call-bind "^1.0.2" - has-symbols "^1.0.1" - -is-resolvable@^1.0.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/is-resolvable/-/is-resolvable-1.1.0.tgz#fb18f87ce1feb925169c9a407c19318a3206ed88" - integrity sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg== - -is-stream@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-1.1.0.tgz#12d4a3dd4e68e0b79ceb8dbc84173ae80d91ca44" - integrity sha1-EtSj3U5o4Lec6428hBc66A2RykQ= - -is-string@^1.0.5: - version "1.0.5" - resolved "https://registry.yarnpkg.com/is-string/-/is-string-1.0.5.tgz#40493ed198ef3ff477b8c7f92f644ec82a5cd3a6" - integrity sha512-buY6VNRjhQMiF1qWDouloZlQbRhDPCebwxSjxMjxgemYT46YMd2NR0/H+fBhEfWX4A/w9TBJ+ol+okqJKFE6vQ== - -is-svg@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/is-svg/-/is-svg-3.0.0.tgz#9321dbd29c212e5ca99c4fa9794c714bcafa2f75" - integrity sha512-gi4iHK53LR2ujhLVVj+37Ykh9GLqYHX6JOVXbLAucaG/Cqw9xwdFOjDM2qeifLs1sF1npXXFvDu0r5HNgCMrzQ== - dependencies: - html-comment-regex "^1.1.0" - -is-symbol@^1.0.2, is-symbol@^1.0.3: - version "1.0.3" - resolved "https://registry.yarnpkg.com/is-symbol/-/is-symbol-1.0.3.tgz#38e1014b9e6329be0de9d24a414fd7441ec61937" - integrity sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ== - dependencies: - has-symbols "^1.0.1" - -is-typedarray@~1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/is-typedarray/-/is-typedarray-1.0.0.tgz#e479c80858df0c1b11ddda6940f96011fcda4a9a" - integrity sha1-5HnICFjfDBsR3dppQPlgEfzaSpo= - -is-windows@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/is-windows/-/is-windows-1.0.2.tgz#d1850eb9791ecd18e6182ce12a30f396634bb19d" - integrity sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA== - -is-wsl@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/is-wsl/-/is-wsl-1.1.0.tgz#1f16e4aa22b04d1336b66188a66af3c600c3a66d" - integrity sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0= - -is-wsl@^2.1.1: - version "2.2.0" - resolved "https://registry.yarnpkg.com/is-wsl/-/is-wsl-2.2.0.tgz#74a4c76e77ca9fd3f932f290c17ea326cd157271" - integrity sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww== - dependencies: - is-docker "^2.0.0" - -isarray@1.0.0, isarray@^1.0.0, isarray@~1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11" - integrity sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE= - -isarray@2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/isarray/-/isarray-2.0.1.tgz#a37d94ed9cda2d59865c9f76fe596ee1f338741e" - integrity sha1-o32U7ZzaLVmGXJ92/llu4fM4dB4= - -isbinaryfile@^4.0.6: - version "4.0.6" - resolved "https://registry.yarnpkg.com/isbinaryfile/-/isbinaryfile-4.0.6.tgz#edcb62b224e2b4710830b67498c8e4e5a4d2610b" - integrity sha512-ORrEy+SNVqUhrCaal4hA4fBzhggQQ+BaLntyPOdoEiwlKZW9BZiJXjg3RMiruE4tPEI3pyVPpySHQF/dKWperg== - -isexe@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" - integrity sha1-6PvzdNxVb/iUehDcsFctYz8s+hA= - -isobject@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/isobject/-/isobject-2.1.0.tgz#f065561096a3f1da2ef46272f815c840d87e0c89" - integrity sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk= - dependencies: - isarray "1.0.0" - -isobject@^3.0.0, isobject@^3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/isobject/-/isobject-3.0.1.tgz#4e431e92b11a9731636aa1f9c8d1ccbcfdab78df" - integrity sha1-TkMekrEalzFjaqH5yNHMvP2reN8= - -isstream@~0.1.2: - version "0.1.2" - resolved "https://registry.yarnpkg.com/isstream/-/isstream-0.1.2.tgz#47e63f7af55afa6f92e1500e690eb8b8529c099a" - integrity sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo= - -istanbul-lib-coverage@^2.0.5: - version "2.0.5" - resolved "https://registry.yarnpkg.com/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.5.tgz#675f0ab69503fad4b1d849f736baaca803344f49" - integrity sha512-8aXznuEPCJvGnMSRft4udDRDtb1V3pkQkMMI5LI+6HuQz5oQ4J2UFn1H82raA3qJtyOLkkwVqICBQkjnGtn5mA== - -istanbul-lib-coverage@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/istanbul-lib-coverage/-/istanbul-lib-coverage-3.0.0.tgz#f5944a37c70b550b02a78a5c3b2055b280cec8ec" - integrity sha512-UiUIqxMgRDET6eR+o5HbfRYP1l0hqkWOs7vNxC/mggutCMUIhWMm8gAHb8tHlyfD3/l6rlgNA5cKdDzEAf6hEg== - -istanbul-lib-instrument@^4.0.3: - version "4.0.3" - resolved "https://registry.yarnpkg.com/istanbul-lib-instrument/-/istanbul-lib-instrument-4.0.3.tgz#873c6fff897450118222774696a3f28902d77c1d" - integrity sha512-BXgQl9kf4WTCPCCpmFGoJkz/+uhvm7h7PFKUYxh7qarQd3ER33vHG//qaE8eN25l07YqZPpHXU9I09l/RD5aGQ== - dependencies: - "@babel/core" "^7.7.5" - "@istanbuljs/schema" "^0.1.2" - istanbul-lib-coverage "^3.0.0" - semver "^6.3.0" - -istanbul-lib-report@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz#7518fe52ea44de372f460a76b5ecda9ffb73d8a6" - integrity sha512-wcdi+uAKzfiGT2abPpKZ0hSU1rGQjUQnLvtY5MpQ7QCTahD3VODhcu4wcfY1YtkGaDD5yuydOLINXsfbus9ROw== - dependencies: - istanbul-lib-coverage "^3.0.0" - make-dir "^3.0.0" - supports-color "^7.1.0" - -istanbul-lib-source-maps@^3.0.6: - version "3.0.6" - resolved "https://registry.yarnpkg.com/istanbul-lib-source-maps/-/istanbul-lib-source-maps-3.0.6.tgz#284997c48211752ec486253da97e3879defba8c8" - integrity sha512-R47KzMtDJH6X4/YW9XTx+jrLnZnscW4VpNN+1PViSYTejLVPWv7oov+Duf8YQSPyVRUvueQqz1TcsC6mooZTXw== - dependencies: - debug "^4.1.1" - istanbul-lib-coverage "^2.0.5" - make-dir "^2.1.0" - rimraf "^2.6.3" - source-map "^0.6.1" - -istanbul-reports@^3.0.2: - version "3.0.2" - resolved "https://registry.yarnpkg.com/istanbul-reports/-/istanbul-reports-3.0.2.tgz#d593210e5000683750cb09fc0644e4b6e27fd53b" - integrity sha512-9tZvz7AiR3PEDNGiV9vIouQ/EAcqMXFmkcA1CDFTwOB98OZVDL0PH9glHotf5Ugp6GCOTypfzGWI/OqjWNCRUw== - dependencies: - html-escaper "^2.0.0" - istanbul-lib-report "^3.0.0" - -jasmine-core@^3.6.0, jasmine-core@~3.6.0: - version "3.6.0" - resolved "https://registry.yarnpkg.com/jasmine-core/-/jasmine-core-3.6.0.tgz#491f3bb23941799c353ceb7a45b38a950ebc5a20" - integrity sha512-8uQYa7zJN8hq9z+g8z1bqCfdC8eoDAeVnM5sfqs7KHv9/ifoJ500m018fpFc7RDaO6SWCLCXwo/wPSNcdYTgcw== - -jasmine-core@~2.8.0: - version "2.8.0" - resolved "https://registry.yarnpkg.com/jasmine-core/-/jasmine-core-2.8.0.tgz#bcc979ae1f9fd05701e45e52e65d3a5d63f1a24e" - integrity sha1-vMl5rh+f0FcB5F5S5l06XWPxok4= - -jasmine-spec-reporter@~5.0.0: - version "5.0.2" - resolved "https://registry.yarnpkg.com/jasmine-spec-reporter/-/jasmine-spec-reporter-5.0.2.tgz#b61288ab074ad440dc2477c4d42840b0e74a6b95" - integrity sha512-6gP1LbVgJ+d7PKksQBc2H0oDGNRQI3gKUsWlswKaQ2fif9X5gzhQcgM5+kiJGCQVurOG09jqNhk7payggyp5+g== - dependencies: - colors "1.4.0" - -jasmine@2.8.0: - version "2.8.0" - resolved "https://registry.yarnpkg.com/jasmine/-/jasmine-2.8.0.tgz#6b089c0a11576b1f16df11b80146d91d4e8b8a3e" - integrity sha1-awicChFXax8W3xG4AUbZHU6Lij4= - dependencies: - exit "^0.1.2" - glob "^7.0.6" - jasmine-core "~2.8.0" - -jasminewd2@^2.1.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/jasminewd2/-/jasminewd2-2.2.0.tgz#e37cf0b17f199cce23bea71b2039395246b4ec4e" - integrity sha1-43zwsX8ZnM4jvqcbIDk5Uka07E4= - -jest-worker@26.5.0: - version "26.5.0" - resolved "https://registry.yarnpkg.com/jest-worker/-/jest-worker-26.5.0.tgz#87deee86dbbc5f98d9919e0dadf2c40e3152fa30" - integrity sha512-kTw66Dn4ZX7WpjZ7T/SUDgRhapFRKWmisVAF0Rv4Fu8SLFD7eLbqpLvbxVqYhSgaWa7I+bW7pHnbyfNsH6stug== - dependencies: - "@types/node" "*" - merge-stream "^2.0.0" - supports-color "^7.0.0" - -jest-worker@^26.5.0: - version "26.6.2" - resolved "https://registry.yarnpkg.com/jest-worker/-/jest-worker-26.6.2.tgz#7f72cbc4d643c365e27b9fd775f9d0eaa9c7a8ed" - integrity sha512-KWYVV1c4i+jbMpaBC+U++4Va0cp8OisU185o73T1vo99hqi7w8tSJfUXYswwqqrjzwxa6KpRK54WhPvwf5w6PQ== - dependencies: - "@types/node" "*" - merge-stream "^2.0.0" - supports-color "^7.0.0" - -js-tokens@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-4.0.0.tgz#19203fb59991df98e3a287050d4647cdeaf32499" - integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ== - -js-yaml@^3.13.1: - version "3.14.1" - resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.14.1.tgz#dae812fdb3825fa306609a8717383c50c36a0537" - integrity sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g== - dependencies: - argparse "^1.0.7" - esprima "^4.0.0" - -jsbn@~0.1.0: - version "0.1.1" - resolved "https://registry.yarnpkg.com/jsbn/-/jsbn-0.1.1.tgz#a5e654c2e5a2deb5f201d96cefbca80c0ef2f513" - integrity sha1-peZUwuWi3rXyAdls77yoDA7y9RM= - -jsesc@^2.5.1: - version "2.5.2" - resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-2.5.2.tgz#80564d2e483dacf6e8ef209650a67df3f0c283a4" - integrity sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA== - -jsesc@~0.5.0: - version "0.5.0" - resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-0.5.0.tgz#e7dee66e35d6fc16f710fe91d5cf69f70f08911d" - integrity sha1-597mbjXW/Bb3EP6R1c9p9w8IkR0= - -json-buffer@3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/json-buffer/-/json-buffer-3.0.0.tgz#5b1f397afc75d677bde8bcfc0e47e1f9a3d9a898" - integrity sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg= - -json-parse-better-errors@^1.0.0, json-parse-better-errors@^1.0.1, json-parse-better-errors@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz#bb867cfb3450e69107c131d1c514bab3dc8bcaa9" - integrity sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw== - -json-parse-even-better-errors@^2.3.0: - version "2.3.1" - resolved "https://registry.yarnpkg.com/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz#7c47805a94319928e05777405dc12e1f7a4ee02d" - integrity sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w== - -json-schema-traverse@^0.4.1: - version "0.4.1" - resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" - integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg== - -json-schema@0.2.3: - version "0.2.3" - resolved "https://registry.yarnpkg.com/json-schema/-/json-schema-0.2.3.tgz#b480c892e59a2f05954ce727bd3f2a4e882f9e13" - integrity sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM= - -json-stringify-safe@^5.0.1, json-stringify-safe@~5.0.1: - version "5.0.1" - resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb" - integrity sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus= - -json3@^3.3.2: - version "3.3.3" - resolved "https://registry.yarnpkg.com/json3/-/json3-3.3.3.tgz#7fc10e375fc5ae42c4705a5cc0aa6f62be305b81" - integrity sha512-c7/8mbUsKigAbLkD5B010BK4D9LZm7A1pNItkEwiUZRpIN66exu/e7YQWysGun+TRKaJp8MhemM+VkfWv42aCA== - -json5@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/json5/-/json5-1.0.1.tgz#779fb0018604fa854eacbf6252180d83543e3dbe" - integrity sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow== - dependencies: - minimist "^1.2.0" - -json5@^2.1.2: - version "2.2.0" - resolved "https://registry.yarnpkg.com/json5/-/json5-2.2.0.tgz#2dfefe720c6ba525d9ebd909950f0515316c89a3" - integrity sha512-f+8cldu7X/y7RAJurMEJmdoKXGB/X550w2Nr3tTbezL6RwEE/iMcm+tZnXeoZtKuOq6ft8+CqzEkrIgx1fPoQA== - dependencies: - minimist "^1.2.5" - -jsonc-parser@2.3.1: - version "2.3.1" - resolved "https://registry.yarnpkg.com/jsonc-parser/-/jsonc-parser-2.3.1.tgz#59549150b133f2efacca48fe9ce1ec0659af2342" - integrity sha512-H8jvkz1O50L3dMZCsLqiuB2tA7muqbSg1AtGEkN0leAqGjsUzDJir3Zwr02BhqdcITPg3ei3mZ+HjMocAknhhg== - -jsonfile@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-4.0.0.tgz#8771aae0799b64076b76640fca058f9c10e33ecb" - integrity sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss= - optionalDependencies: - graceful-fs "^4.1.6" - -jsonparse@^1.2.0: - version "1.3.1" - resolved "https://registry.yarnpkg.com/jsonparse/-/jsonparse-1.3.1.tgz#3f4dae4a91fac315f71062f8521cc239f1366280" - integrity sha1-P02uSpH6wxX3EGL4UhzCOfE2YoA= - -jsprim@^1.2.2: - version "1.4.1" - resolved "https://registry.yarnpkg.com/jsprim/-/jsprim-1.4.1.tgz#313e66bc1e5cc06e438bc1b7499c2e5c56acb6a2" - integrity sha1-MT5mvB5cwG5Di8G3SZwuXFastqI= - dependencies: - assert-plus "1.0.0" - extsprintf "1.3.0" - json-schema "0.2.3" - verror "1.10.0" - -jszip@^3.1.3: - version "3.6.0" - resolved "https://registry.yarnpkg.com/jszip/-/jszip-3.6.0.tgz#839b72812e3f97819cc13ac4134ffced95dd6af9" - integrity sha512-jgnQoG9LKnWO3mnVNBnfhkh0QknICd1FGSrXcgrl67zioyJ4wgx25o9ZqwNtrROSflGBCGYnJfjrIyRIby1OoQ== - dependencies: - lie "~3.3.0" - pako "~1.0.2" - readable-stream "~2.3.6" - set-immediate-shim "~1.0.1" - -karma-chrome-launcher@~3.1.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/karma-chrome-launcher/-/karma-chrome-launcher-3.1.0.tgz#805a586799a4d05f4e54f72a204979f3f3066738" - integrity sha512-3dPs/n7vgz1rxxtynpzZTvb9y/GIaW8xjAwcIGttLbycqoFtI7yo1NGnQi6oFTherRE+GIhCAHZC4vEqWGhNvg== - dependencies: - which "^1.2.1" - -karma-coverage-istanbul-reporter@~3.0.2: - version "3.0.3" - resolved "https://registry.yarnpkg.com/karma-coverage-istanbul-reporter/-/karma-coverage-istanbul-reporter-3.0.3.tgz#f3b5303553aadc8e681d40d360dfdc19bc7e9fe9" - integrity sha512-wE4VFhG/QZv2Y4CdAYWDbMmcAHeS926ZIji4z+FkB2aF/EposRb6DP6G5ncT/wXhqUfAb/d7kZrNKPonbvsATw== - dependencies: - istanbul-lib-coverage "^3.0.0" - istanbul-lib-report "^3.0.0" - istanbul-lib-source-maps "^3.0.6" - istanbul-reports "^3.0.2" - minimatch "^3.0.4" - -karma-jasmine-html-reporter@^1.5.0: - version "1.5.4" - resolved "https://registry.yarnpkg.com/karma-jasmine-html-reporter/-/karma-jasmine-html-reporter-1.5.4.tgz#669f33d694d88fce1b0ccfda57111de716cb0192" - integrity sha512-PtilRLno5O6wH3lDihRnz0Ba8oSn0YUJqKjjux1peoYGwo0AQqrWRbdWk/RLzcGlb+onTyXAnHl6M+Hu3UxG/Q== - -karma-jasmine@~4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/karma-jasmine/-/karma-jasmine-4.0.1.tgz#b99e073b6d99a5196fc4bffc121b89313b0abd82" - integrity sha512-h8XDAhTiZjJKzfkoO1laMH+zfNlra+dEQHUAjpn5JV1zCPtOIVWGQjLBrqhnzQa/hrU2XrZwSyBa6XjEBzfXzw== - dependencies: - jasmine-core "^3.6.0" - -karma-source-map-support@1.4.0: - version "1.4.0" - resolved "https://registry.yarnpkg.com/karma-source-map-support/-/karma-source-map-support-1.4.0.tgz#58526ceccf7e8730e56effd97a4de8d712ac0d6b" - integrity sha512-RsBECncGO17KAoJCYXjv+ckIz+Ii9NCi+9enk+rq6XC81ezYkb4/RHE6CTXdA7IOJqoF3wcaLfVG0CPmE5ca6A== - dependencies: - source-map-support "^0.5.5" - -karma@~5.1.1: - version "5.1.1" - resolved "https://registry.yarnpkg.com/karma/-/karma-5.1.1.tgz#4e472c1e5352d73edbd2090726afdb01d7869d72" - integrity sha512-xAlOr5PMqUbiKXSv5PCniHWV3aiwj6wIZ0gUVcwpTCPVQm/qH2WAMFWxtnpM6KJqhkRWrIpovR4Rb0rn8GtJzQ== - dependencies: - body-parser "^1.19.0" - braces "^3.0.2" - chokidar "^3.0.0" - colors "^1.4.0" - connect "^3.7.0" - di "^0.0.1" - dom-serialize "^2.2.1" - flatted "^2.0.2" - glob "^7.1.6" - graceful-fs "^4.2.4" - http-proxy "^1.18.1" - isbinaryfile "^4.0.6" - lodash "^4.17.15" - log4js "^6.2.1" - mime "^2.4.5" - minimatch "^3.0.4" - qjobs "^1.2.0" - range-parser "^1.2.1" - rimraf "^3.0.2" - socket.io "^2.3.0" - source-map "^0.6.1" - tmp "0.2.1" - ua-parser-js "0.7.21" - yargs "^15.3.1" - -keyv@^3.0.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/keyv/-/keyv-3.1.0.tgz#ecc228486f69991e49e9476485a5be1e8fc5c4d9" - integrity sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA== - dependencies: - json-buffer "3.0.0" - -killable@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/killable/-/killable-1.0.1.tgz#4c8ce441187a061c7474fb87ca08e2a638194892" - integrity sha512-LzqtLKlUwirEUyl/nicirVmNiPvYs7l5n8wOPP7fyJVpUPkvCnW/vuiXGpylGUlnPDnB7311rARzAt3Mhswpjg== - -kind-of@^3.0.2, kind-of@^3.0.3, kind-of@^3.2.0: - version "3.2.2" - resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-3.2.2.tgz#31ea21a734bab9bbb0f32466d893aea51e4a3c64" - integrity sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ= - dependencies: - is-buffer "^1.1.5" - -kind-of@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-4.0.0.tgz#20813df3d712928b207378691a45066fae72dd57" - integrity sha1-IIE989cSkosgc3hpGkUGb65y3Vc= - dependencies: - is-buffer "^1.1.5" - -kind-of@^5.0.0: - version "5.1.0" - resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-5.1.0.tgz#729c91e2d857b7a419a1f9aa65685c4c33f5845d" - integrity sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw== - -kind-of@^6.0.0, kind-of@^6.0.2: - version "6.0.3" - resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-6.0.3.tgz#07c05034a6c349fa06e24fa35aa76db4580ce4dd" - integrity sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw== - -klona@^2.0.4: - version "2.0.4" - resolved "https://registry.yarnpkg.com/klona/-/klona-2.0.4.tgz#7bb1e3affb0cb8624547ef7e8f6708ea2e39dfc0" - integrity sha512-ZRbnvdg/NxqzC7L9Uyqzf4psi1OM4Cuc+sJAkQPjO6XkQIJTNbfK2Rsmbw8fx1p2mkZdp2FZYo2+LwXYY/uwIA== - -less-loader@7.0.2: - version "7.0.2" - resolved "https://registry.yarnpkg.com/less-loader/-/less-loader-7.0.2.tgz#0d73a49ec32a9d3ff12614598e6e2b47fb2a35c4" - integrity sha512-7MKlgjnkCf63E3Lv6w2FvAEgLMx3d/tNBExITcanAq7ys5U8VPWT3F6xcRjYmdNfkoQ9udoVFb1r2azSiTnD6w== - dependencies: - klona "^2.0.4" - loader-utils "^2.0.0" - schema-utils "^3.0.0" - -less@3.12.2: - version "3.12.2" - resolved "https://registry.yarnpkg.com/less/-/less-3.12.2.tgz#157e6dd32a68869df8859314ad38e70211af3ab4" - integrity sha512-+1V2PCMFkL+OIj2/HrtrvZw0BC0sYLMICJfbQjuj/K8CEnlrFX6R5cKKgzzttsZDHyxQNL1jqMREjKN3ja/E3Q== - dependencies: - tslib "^1.10.0" - optionalDependencies: - errno "^0.1.1" - graceful-fs "^4.1.2" - image-size "~0.5.0" - make-dir "^2.1.0" - mime "^1.4.1" - native-request "^1.0.5" - source-map "~0.6.0" - -license-webpack-plugin@2.3.1: - version "2.3.1" - resolved "https://registry.yarnpkg.com/license-webpack-plugin/-/license-webpack-plugin-2.3.1.tgz#08eddb2f776c7c64c02f308a00e017d6e824d0b6" - integrity sha512-yhqTmlYIEpZWA122lf6E0G8+rkn0AzoQ1OpzUKKs/lXUqG1plmGnwmkuuPlfggzJR5y6DLOdot/Tv00CC51CeQ== - dependencies: - "@types/webpack-sources" "^0.1.5" - webpack-sources "^1.2.0" - -lie@~3.3.0: - version "3.3.0" - resolved "https://registry.yarnpkg.com/lie/-/lie-3.3.0.tgz#dcf82dee545f46074daf200c7c1c5a08e0f40f6a" - integrity sha512-UaiMJzeWRlEujzAuw5LokY1L5ecNQYZKfmyZ9L7wDHb/p5etKaxXhohBcrw0EYby+G/NA52vRSN4N39dxHAIwQ== - dependencies: - immediate "~3.0.5" - -lines-and-columns@^1.1.6: - version "1.1.6" - resolved "https://registry.yarnpkg.com/lines-and-columns/-/lines-and-columns-1.1.6.tgz#1c00c743b433cd0a4e80758f7b64a57440d9ff00" - integrity sha1-HADHQ7QzzQpOgHWPe2SldEDZ/wA= - -loader-runner@^2.4.0: - version "2.4.0" - resolved "https://registry.yarnpkg.com/loader-runner/-/loader-runner-2.4.0.tgz#ed47066bfe534d7e84c4c7b9998c2a75607d9357" - integrity sha512-Jsmr89RcXGIwivFY21FcRrisYZfvLMTWx5kOLc+JTxtpBOG6xML0vzbc6SEQG2FO9/4Fc3wW4LVcB5DmGflaRw== - -loader-utils@1.2.3: - version "1.2.3" - resolved "https://registry.yarnpkg.com/loader-utils/-/loader-utils-1.2.3.tgz#1ff5dc6911c9f0a062531a4c04b609406108c2c7" - integrity sha512-fkpz8ejdnEMG3s37wGL07iSBDg99O9D5yflE9RGNH3hRdx9SOwYfnGYdZOUIZitN8E+E2vkq3MUMYMvPYl5ZZA== - dependencies: - big.js "^5.2.2" - emojis-list "^2.0.0" - json5 "^1.0.1" - -loader-utils@2.0.0, loader-utils@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/loader-utils/-/loader-utils-2.0.0.tgz#e4cace5b816d425a166b5f097e10cd12b36064b0" - integrity sha512-rP4F0h2RaWSvPEkD7BLDFQnvSf+nK+wr3ESUjNTyAGobqrijmW92zc+SO6d4p4B1wh7+B/Jg1mkQe5NYUEHtHQ== - dependencies: - big.js "^5.2.2" - emojis-list "^3.0.0" - json5 "^2.1.2" - -loader-utils@^1.1.0, loader-utils@^1.2.3, loader-utils@^1.4.0: - version "1.4.0" - resolved "https://registry.yarnpkg.com/loader-utils/-/loader-utils-1.4.0.tgz#c579b5e34cb34b1a74edc6c1fb36bfa371d5a613" - integrity sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA== - dependencies: - big.js "^5.2.2" - emojis-list "^3.0.0" - json5 "^1.0.1" - -locate-path@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-3.0.0.tgz#dbec3b3ab759758071b58fe59fc41871af21400e" - integrity sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A== - dependencies: - p-locate "^3.0.0" - path-exists "^3.0.0" - -locate-path@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-5.0.0.tgz#1afba396afd676a6d42504d0a67a3a7eb9f62aa0" - integrity sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g== - dependencies: - p-locate "^4.1.0" - -lodash.memoize@^4.1.2: - version "4.1.2" - resolved "https://registry.yarnpkg.com/lodash.memoize/-/lodash.memoize-4.1.2.tgz#bcc6c49a42a2840ed997f323eada5ecd182e0bfe" - integrity sha1-vMbEmkKihA7Zl/Mj6tpezRguC/4= - -lodash.uniq@^4.5.0: - version "4.5.0" - resolved "https://registry.yarnpkg.com/lodash.uniq/-/lodash.uniq-4.5.0.tgz#d0225373aeb652adc1bc82e4945339a842754773" - integrity sha1-0CJTc662Uq3BvILklFM5qEJ1R3M= - -lodash@^4.17.10, lodash@^4.17.11, lodash@^4.17.14, lodash@^4.17.15, lodash@^4.17.19: - version "4.17.21" - resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c" - integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg== - -log-symbols@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/log-symbols/-/log-symbols-4.0.0.tgz#69b3cc46d20f448eccdb75ea1fa733d9e821c920" - integrity sha512-FN8JBzLx6CzeMrB0tg6pqlGU1wCrXW+ZXGH481kfsBqer0hToTIiHdjH4Mq8xJUbvATujKCvaREGWpGUionraA== - dependencies: - chalk "^4.0.0" - -log4js@^6.2.1: - version "6.3.0" - resolved "https://registry.yarnpkg.com/log4js/-/log4js-6.3.0.tgz#10dfafbb434351a3e30277a00b9879446f715bcb" - integrity sha512-Mc8jNuSFImQUIateBFwdOQcmC6Q5maU0VVvdC2R6XMb66/VnT+7WS4D/0EeNMZu1YODmJe5NIn2XftCzEocUgw== - dependencies: - date-format "^3.0.0" - debug "^4.1.1" - flatted "^2.0.1" - rfdc "^1.1.4" - streamroller "^2.2.4" - -loglevel@^1.6.8: - version "1.7.1" - resolved "https://registry.yarnpkg.com/loglevel/-/loglevel-1.7.1.tgz#005fde2f5e6e47068f935ff28573e125ef72f197" - integrity sha512-Hesni4s5UkWkwCGJMQGAh71PaLUmKFM60dHvq0zi/vDhhrzuk+4GgNbTXJ12YYQJn6ZKBDNIjYcuQGKudvqrIw== - -lowercase-keys@^1.0.0, lowercase-keys@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-1.0.1.tgz#6f9e30b47084d971a7c820ff15a6c5167b74c26f" - integrity sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA== - -lowercase-keys@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-2.0.0.tgz#2603e78b7b4b0006cbca2fbcc8a3202558ac9479" - integrity sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA== - -lru-cache@^5.1.1: - version "5.1.1" - resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-5.1.1.tgz#1da27e6710271947695daf6848e847f01d84b920" - integrity sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w== - dependencies: - yallist "^3.0.2" - -lru-cache@^6.0.0: - version "6.0.0" - resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94" - integrity sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA== - dependencies: - yallist "^4.0.0" - -magic-string@0.25.7, magic-string@^0.25.0: - version "0.25.7" - resolved "https://registry.yarnpkg.com/magic-string/-/magic-string-0.25.7.tgz#3f497d6fd34c669c6798dcb821f2ef31f5445051" - integrity sha512-4CrMT5DOHTDk4HYDlzmwu4FVCcIYI8gauveasrdCu2IKIFOJ3f0v/8MDGJCDL9oD2ppz/Av1b0Nj345H9M+XIA== - dependencies: - sourcemap-codec "^1.4.4" - -make-dir@^2.0.0, make-dir@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-2.1.0.tgz#5f0310e18b8be898cc07009295a30ae41e91e6f5" - integrity sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA== - dependencies: - pify "^4.0.1" - semver "^5.6.0" - -make-dir@^3.0.0, make-dir@^3.0.2: - version "3.1.0" - resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-3.1.0.tgz#415e967046b3a7f1d185277d84aa58203726a13f" - integrity sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw== - dependencies: - semver "^6.0.0" - -make-error@^1.1.1: - version "1.3.6" - resolved "https://registry.yarnpkg.com/make-error/-/make-error-1.3.6.tgz#2eb2e37ea9b67c4891f684a1394799af484cf7a2" - integrity sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw== - -make-fetch-happen@^5.0.0: - version "5.0.2" - resolved "https://registry.yarnpkg.com/make-fetch-happen/-/make-fetch-happen-5.0.2.tgz#aa8387104f2687edca01c8687ee45013d02d19bd" - integrity sha512-07JHC0r1ykIoruKO8ifMXu+xEU8qOXDFETylktdug6vJDACnP+HKevOu3PXyNPzFyTSlz8vrBYlBO1JZRe8Cag== - dependencies: - agentkeepalive "^3.4.1" - cacache "^12.0.0" - http-cache-semantics "^3.8.1" - http-proxy-agent "^2.1.0" - https-proxy-agent "^2.2.3" - lru-cache "^5.1.1" - mississippi "^3.0.0" - node-fetch-npm "^2.0.2" - promise-retry "^1.1.1" - socks-proxy-agent "^4.0.0" - ssri "^6.0.0" - -map-cache@^0.2.2: - version "0.2.2" - resolved "https://registry.yarnpkg.com/map-cache/-/map-cache-0.2.2.tgz#c32abd0bd6525d9b051645bb4f26ac5dc98a0dbf" - integrity sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8= - -map-visit@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/map-visit/-/map-visit-1.0.0.tgz#ecdca8f13144e660f1b5bd41f12f3479d98dfb8f" - integrity sha1-7Nyo8TFE5mDxtb1B8S80edmN+48= - dependencies: - object-visit "^1.0.0" - -matcher@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/matcher/-/matcher-3.0.0.tgz#bd9060f4c5b70aa8041ccc6f80368760994f30ca" - integrity sha512-OkeDaAZ/bQCxeFAozM55PKcKU0yJMPGifLwV4Qgjitu+5MoAfSQN4lsLJeXZ1b8w0x+/Emda6MZgXS1jvsapng== - dependencies: - escape-string-regexp "^4.0.0" - -md5.js@^1.3.4: - version "1.3.5" - resolved "https://registry.yarnpkg.com/md5.js/-/md5.js-1.3.5.tgz#b5d07b8e3216e3e27cd728d72f70d1e6a342005f" - integrity sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg== - dependencies: - hash-base "^3.0.0" - inherits "^2.0.1" - safe-buffer "^5.1.2" - -mdn-data@2.0.14: - version "2.0.14" - resolved "https://registry.yarnpkg.com/mdn-data/-/mdn-data-2.0.14.tgz#7113fc4281917d63ce29b43446f701e68c25ba50" - integrity sha512-dn6wd0uw5GsdswPFfsgMp5NSB0/aDe6fK94YJV/AJDYXL6HVLWBsxeq7js7Ad+mU2K9LAlwpk6kN2D5mwCPVow== - -mdn-data@2.0.4: - version "2.0.4" - resolved "https://registry.yarnpkg.com/mdn-data/-/mdn-data-2.0.4.tgz#699b3c38ac6f1d728091a64650b65d388502fd5b" - integrity sha512-iV3XNKw06j5Q7mi6h+9vbx23Tv7JkjEVgKHW4pimwyDGWm0OIQntJJ+u1C6mg6mK1EaTv42XQ7w76yuzH7M2cA== - -media-typer@0.3.0: - version "0.3.0" - resolved "https://registry.yarnpkg.com/media-typer/-/media-typer-0.3.0.tgz#8710d7af0aa626f8fffa1ce00168545263255748" - integrity sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g= - -memory-fs@^0.4.1: - version "0.4.1" - resolved "https://registry.yarnpkg.com/memory-fs/-/memory-fs-0.4.1.tgz#3a9a20b8462523e447cfbc7e8bb80ed667bfc552" - integrity sha1-OpoguEYlI+RHz7x+i7gO1me/xVI= - dependencies: - errno "^0.1.3" - readable-stream "^2.0.1" - -memory-fs@^0.5.0: - version "0.5.0" - resolved "https://registry.yarnpkg.com/memory-fs/-/memory-fs-0.5.0.tgz#324c01288b88652966d161db77838720845a8e3c" - integrity sha512-jA0rdU5KoQMC0e6ppoNRtpp6vjFq6+NY7r8hywnC7V+1Xj/MtHwGIbB1QaK/dunyjWteJzmkpd7ooeWg10T7GA== - dependencies: - errno "^0.1.3" - readable-stream "^2.0.1" - -merge-descriptors@1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/merge-descriptors/-/merge-descriptors-1.0.1.tgz#b00aaa556dd8b44568150ec9d1b953f3f90cbb61" - integrity sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E= - -merge-source-map@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/merge-source-map/-/merge-source-map-1.1.0.tgz#2fdde7e6020939f70906a68f2d7ae685e4c8c646" - integrity sha512-Qkcp7P2ygktpMPh2mCQZaf3jhN6D3Z/qVZHSdWvQ+2Ef5HgRAPBO57A77+ENm0CPx2+1Ce/MYKi3ymqdfuqibw== - dependencies: - source-map "^0.6.1" - -merge-stream@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60" - integrity sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w== - -merge2@^1.3.0: - version "1.4.1" - resolved "https://registry.yarnpkg.com/merge2/-/merge2-1.4.1.tgz#4368892f885e907455a6fd7dc55c0c9d404990ae" - integrity sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg== - -methods@~1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/methods/-/methods-1.1.2.tgz#5529a4d67654134edcc5266656835b0f851afcee" - integrity sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4= - -micromatch@^3.1.10, micromatch@^3.1.4: - version "3.1.10" - resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-3.1.10.tgz#70859bc95c9840952f359a068a3fc49f9ecfac23" - integrity sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg== - dependencies: - arr-diff "^4.0.0" - array-unique "^0.3.2" - braces "^2.3.1" - define-property "^2.0.2" - extend-shallow "^3.0.2" - extglob "^2.0.4" - fragment-cache "^0.2.1" - kind-of "^6.0.2" - nanomatch "^1.2.9" - object.pick "^1.3.0" - regex-not "^1.0.0" - snapdragon "^0.8.1" - to-regex "^3.0.2" - -micromatch@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.2.tgz#4fcb0999bf9fbc2fcbdd212f6d629b9a56c39259" - integrity sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q== - dependencies: - braces "^3.0.1" - picomatch "^2.0.5" - -miller-rabin@^4.0.0: - version "4.0.1" - resolved "https://registry.yarnpkg.com/miller-rabin/-/miller-rabin-4.0.1.tgz#f080351c865b0dc562a8462966daa53543c78a4d" - integrity sha512-115fLhvZVqWwHPbClyntxEVfVDfl9DLLTuJvq3g2O/Oxi8AiNouAHvDSzHS0viUJc+V5vm3eq91Xwqn9dp4jRA== - dependencies: - bn.js "^4.0.0" - brorand "^1.0.1" - -mime-db@1.46.0, "mime-db@>= 1.43.0 < 2": - version "1.46.0" - resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.46.0.tgz#6267748a7f799594de3cbc8cde91def349661cee" - integrity sha512-svXaP8UQRZ5K7or+ZmfNhg2xX3yKDMUzqadsSqi4NCH/KomcH75MAMYAGVlvXn4+b/xOPhS3I2uHKRUzvjY7BQ== - -mime-types@^2.1.12, mime-types@~2.1.17, mime-types@~2.1.19, mime-types@~2.1.24: - version "2.1.29" - resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.29.tgz#1d4ab77da64b91f5f72489df29236563754bb1b2" - integrity sha512-Y/jMt/S5sR9OaqteJtslsFZKWOIIqMACsJSiHghlCAyhf7jfVYjKBmLiX8OgpWeW+fjJ2b+Az69aPFPkUOY6xQ== - dependencies: - mime-db "1.46.0" - -mime@1.6.0, mime@^1.4.1: - version "1.6.0" - resolved "https://registry.yarnpkg.com/mime/-/mime-1.6.0.tgz#32cd9e5c64553bd58d19a568af452acff04981b1" - integrity sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg== - -mime@^2.4.4, mime@^2.4.5: - version "2.5.2" - resolved "https://registry.yarnpkg.com/mime/-/mime-2.5.2.tgz#6e3dc6cc2b9510643830e5f19d5cb753da5eeabe" - integrity sha512-tqkh47FzKeCPD2PUiPB6pkbMzsCasjxAfC62/Wap5qrUWcb+sFasXUC5I3gYM5iBM8v/Qpn4UK0x+j0iHyFPDg== - -mimic-fn@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b" - integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg== - -mimic-response@^1.0.0, mimic-response@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-1.0.1.tgz#4923538878eef42063cb8a3e3b0798781487ab1b" - integrity sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ== - -mini-css-extract-plugin@1.2.1: - version "1.2.1" - resolved "https://registry.yarnpkg.com/mini-css-extract-plugin/-/mini-css-extract-plugin-1.2.1.tgz#30ea7dee632b3002b0c77aeed447790408cb247e" - integrity sha512-G3yw7/TQaPfkuiR73MDcyiqhyP8SnbmLhUbpC76H+wtQxA6wfKhMCQOCb6wnPK0dQbjORAeOILQqEesg4/wF7A== - dependencies: - loader-utils "^2.0.0" - schema-utils "^3.0.0" - webpack-sources "^1.1.0" - -minimalistic-assert@^1.0.0, minimalistic-assert@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz#2e194de044626d4a10e7f7fbc00ce73e83e4d5c7" - integrity sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A== - -minimalistic-crypto-utils@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz#f6c00c1c0b082246e5c4d99dfb8c7c083b2b582a" - integrity sha1-9sAMHAsIIkblxNmd+4x8CDsrWCo= - -minimatch@3.0.4, minimatch@^3.0.4: - version "3.0.4" - resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.0.4.tgz#5166e286457f03306064be5497e8dbb0c3d32083" - integrity sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA== - dependencies: - brace-expansion "^1.1.7" - -minimist@^1.2.0, minimist@^1.2.5: - version "1.2.5" - resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.5.tgz#67d66014b66a6a8aaa0c083c5fd58df4e4e97602" - integrity sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw== - -minipass-collect@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/minipass-collect/-/minipass-collect-1.0.2.tgz#22b813bf745dc6edba2576b940022ad6edc8c617" - integrity sha512-6T6lH0H8OG9kITm/Jm6tdooIbogG9e0tLgpY6mphXSm/A9u8Nq1ryBG+Qspiub9LjWlBPsPS3tWQ/Botq4FdxA== - dependencies: - minipass "^3.0.0" - -minipass-flush@^1.0.5: - version "1.0.5" - resolved "https://registry.yarnpkg.com/minipass-flush/-/minipass-flush-1.0.5.tgz#82e7135d7e89a50ffe64610a787953c4c4cbb373" - integrity sha512-JmQSYYpPUqX5Jyn1mXaRwOda1uQ8HP5KAT/oDSLCzt1BYRhQU0/hDtsB1ufZfEEzMZ9aAVmsBw8+FWsIXlClWw== - dependencies: - minipass "^3.0.0" - -minipass-pipeline@^1.2.2: - version "1.2.4" - resolved "https://registry.yarnpkg.com/minipass-pipeline/-/minipass-pipeline-1.2.4.tgz#68472f79711c084657c067c5c6ad93cddea8214c" - integrity sha512-xuIq7cIOt09RPRJ19gdi4b+RiNvDFYe5JH+ggNvBqGqpQXcru3PcRmOZuHBKWK1Txf9+cQ+HMVN4d6z46LZP7A== - dependencies: - minipass "^3.0.0" - -minipass@^2.3.5, minipass@^2.6.0, minipass@^2.8.6, minipass@^2.9.0: - version "2.9.0" - resolved "https://registry.yarnpkg.com/minipass/-/minipass-2.9.0.tgz#e713762e7d3e32fed803115cf93e04bca9fcc9a6" - integrity sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg== - dependencies: - safe-buffer "^5.1.2" - yallist "^3.0.0" - -minipass@^3.0.0, minipass@^3.1.1: - version "3.1.3" - resolved "https://registry.yarnpkg.com/minipass/-/minipass-3.1.3.tgz#7d42ff1f39635482e15f9cdb53184deebd5815fd" - integrity sha512-Mgd2GdMVzY+x3IJ+oHnVM+KG3lA5c8tnabyJKmHSaG2kAGpudxuOf8ToDkhumF7UzME7DecbQE9uOZhNm7PuJg== - dependencies: - yallist "^4.0.0" - -minizlib@^1.2.1: - version "1.3.3" - resolved "https://registry.yarnpkg.com/minizlib/-/minizlib-1.3.3.tgz#2290de96818a34c29551c8a8d301216bd65a861d" - integrity sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q== - dependencies: - minipass "^2.9.0" - -minizlib@^2.1.1: - version "2.1.2" - resolved "https://registry.yarnpkg.com/minizlib/-/minizlib-2.1.2.tgz#e90d3466ba209b932451508a11ce3d3632145931" - integrity sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg== - dependencies: - minipass "^3.0.0" - yallist "^4.0.0" - -mississippi@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/mississippi/-/mississippi-3.0.0.tgz#ea0a3291f97e0b5e8776b363d5f0a12d94c67022" - integrity sha512-x471SsVjUtBRtcvd4BzKE9kFC+/2TeWgKCgw0bZcw1b9l2X3QX5vCWgF+KaZaYm87Ss//rHnWryupDrgLvmSkA== - dependencies: - concat-stream "^1.5.0" - duplexify "^3.4.2" - end-of-stream "^1.1.0" - flush-write-stream "^1.0.0" - from2 "^2.1.0" - parallel-transform "^1.1.0" - pump "^3.0.0" - pumpify "^1.3.3" - stream-each "^1.1.0" - through2 "^2.0.0" - -mixin-deep@^1.2.0: - version "1.3.2" - resolved "https://registry.yarnpkg.com/mixin-deep/-/mixin-deep-1.3.2.tgz#1120b43dc359a785dce65b55b82e257ccf479566" - integrity sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA== - dependencies: - for-in "^1.0.2" - is-extendable "^1.0.1" - -mkdirp@^0.5.0, mkdirp@^0.5.1, mkdirp@^0.5.3, mkdirp@^0.5.4, mkdirp@^0.5.5, mkdirp@~0.5.1: - version "0.5.5" - resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.5.tgz#d91cefd62d1436ca0f41620e251288d420099def" - integrity sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ== - dependencies: - minimist "^1.2.5" - -mkdirp@^1.0.3, mkdirp@^1.0.4, mkdirp@~1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-1.0.4.tgz#3eb5ed62622756d79a5f0e2a221dfebad75c2f7e" - integrity sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw== - -move-concurrently@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/move-concurrently/-/move-concurrently-1.0.1.tgz#be2c005fda32e0b29af1f05d7c4b33214c701f92" - integrity sha1-viwAX9oy4LKa8fBdfEszIUxwH5I= - dependencies: - aproba "^1.1.1" - copy-concurrently "^1.0.0" - fs-write-stream-atomic "^1.0.8" - mkdirp "^0.5.1" - rimraf "^2.5.4" - run-queue "^1.0.3" - -ms@2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8" - integrity sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g= - -ms@2.1.1: - version "2.1.1" - resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.1.tgz#30a5864eb3ebb0a66f2ebe6d727af06a09d86e0a" - integrity sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg== - -ms@2.1.2: - version "2.1.2" - resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009" - integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w== - -ms@^2.0.0, ms@^2.1.1: - version "2.1.3" - resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" - integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== - -multicast-dns-service-types@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/multicast-dns-service-types/-/multicast-dns-service-types-1.1.0.tgz#899f11d9686e5e05cb91b35d5f0e63b773cfc901" - integrity sha1-iZ8R2WhuXgXLkbNdXw5jt3PPyQE= - -multicast-dns@^6.0.1: - version "6.2.3" - resolved "https://registry.yarnpkg.com/multicast-dns/-/multicast-dns-6.2.3.tgz#a0ec7bd9055c4282f790c3c82f4e28db3b31b229" - integrity sha512-ji6J5enbMyGRHIAkAOu3WdV8nggqviKCEKtXcOqfphZZtQrmHKycfynJ2V7eVPUA4NhJ6V7Wf4TmGbTwKE9B6g== - dependencies: - dns-packet "^1.3.1" - thunky "^1.0.2" - -mute-stream@0.0.8: - version "0.0.8" - resolved "https://registry.yarnpkg.com/mute-stream/-/mute-stream-0.0.8.tgz#1630c42b2251ff81e2a283de96a5497ea92e5e0d" - integrity sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA== - -nan@^2.12.1: - version "2.14.2" - resolved "https://registry.yarnpkg.com/nan/-/nan-2.14.2.tgz#f5376400695168f4cc694ac9393d0c9585eeea19" - integrity sha512-M2ufzIiINKCuDfBSAUr1vWQ+vuVcA9kqx8JJUsbQi6yf1uGRyb7HfpdfUr5qLXf3B/t8dPvcjhKMmlfnP47EzQ== - -nanomatch@^1.2.9: - version "1.2.13" - resolved "https://registry.yarnpkg.com/nanomatch/-/nanomatch-1.2.13.tgz#b87a8aa4fc0de8fe6be88895b38983ff265bd119" - integrity sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA== - dependencies: - arr-diff "^4.0.0" - array-unique "^0.3.2" - define-property "^2.0.2" - extend-shallow "^3.0.2" - fragment-cache "^0.2.1" - is-windows "^1.0.2" - kind-of "^6.0.2" - object.pick "^1.3.0" - regex-not "^1.0.0" - snapdragon "^0.8.1" - to-regex "^3.0.1" - -native-request@^1.0.5: - version "1.0.8" - resolved "https://registry.yarnpkg.com/native-request/-/native-request-1.0.8.tgz#8f66bf606e0f7ea27c0e5995eb2f5d03e33ae6fb" - integrity sha512-vU2JojJVelUGp6jRcLwToPoWGxSx23z/0iX+I77J3Ht17rf2INGjrhOoQnjVo60nQd8wVsgzKkPfRXBiVdD2ag== - -negotiator@0.6.2: - version "0.6.2" - resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.2.tgz#feacf7ccf525a77ae9634436a64883ffeca346fb" - integrity sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw== - -neo-async@^2.5.0, neo-async@^2.6.1, neo-async@^2.6.2: - version "2.6.2" - resolved "https://registry.yarnpkg.com/neo-async/-/neo-async-2.6.2.tgz#b4aafb93e3aeb2d8174ca53cf163ab7d7308305f" - integrity sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw== - -next-tick@~1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/next-tick/-/next-tick-1.0.0.tgz#ca86d1fe8828169b0120208e3dc8424b9db8342c" - integrity sha1-yobR/ogoFpsBICCOPchCS524NCw= - -ngx-electron@2.2.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/ngx-electron/-/ngx-electron-2.2.0.tgz#8ab65427476153b499d26f332fa5ca88714d8477" - integrity sha512-Yl7Dsnvp97k0XpIuiB54X7Ij2+zU5x0pCAYnN//VZ9tF7c6S3//OZ9dN9Et7p/zIjCV3Hg9vyw68dPJEZGk+LQ== - dependencies: - tslib "^1.9.0" - -nice-try@^1.0.4: - version "1.0.5" - resolved "https://registry.yarnpkg.com/nice-try/-/nice-try-1.0.5.tgz#a3378a7696ce7d223e88fc9b764bd7ef1089e366" - integrity sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ== - -node-fetch-npm@^2.0.2: - version "2.0.4" - resolved "https://registry.yarnpkg.com/node-fetch-npm/-/node-fetch-npm-2.0.4.tgz#6507d0e17a9ec0be3bec516958a497cec54bf5a4" - integrity sha512-iOuIQDWDyjhv9qSDrj9aq/klt6F9z1p2otB3AV7v3zBDcL/x+OfGsvGQZZCcMZbUf4Ujw1xGNQkjvGnVT22cKg== - dependencies: - encoding "^0.1.11" - json-parse-better-errors "^1.0.0" - safe-buffer "^5.1.1" - -node-forge@^0.10.0: - version "0.10.0" - resolved "https://registry.yarnpkg.com/node-forge/-/node-forge-0.10.0.tgz#32dea2afb3e9926f02ee5ce8794902691a676bf3" - integrity sha512-PPmu8eEeG9saEUvI97fm4OYxXVB6bFvyNTyiUOBichBpFG8A1Ljw3bY62+5oOjDEMHRnd0Y7HQ+x7uzxOzC6JA== - -node-libs-browser@^2.2.1: - version "2.2.1" - resolved "https://registry.yarnpkg.com/node-libs-browser/-/node-libs-browser-2.2.1.tgz#b64f513d18338625f90346d27b0d235e631f6425" - integrity sha512-h/zcD8H9kaDZ9ALUWwlBUDo6TKF8a7qBSCSEGfjTVIYeqsioSKaAX+BN7NgiMGp6iSIXZ3PxgCu8KS3b71YK5Q== - dependencies: - assert "^1.1.1" - browserify-zlib "^0.2.0" - buffer "^4.3.0" - console-browserify "^1.1.0" - constants-browserify "^1.0.0" - crypto-browserify "^3.11.0" - domain-browser "^1.1.1" - events "^3.0.0" - https-browserify "^1.0.0" - os-browserify "^0.3.0" - path-browserify "0.0.1" - process "^0.11.10" - punycode "^1.2.4" - querystring-es3 "^0.2.0" - readable-stream "^2.3.3" - stream-browserify "^2.0.1" - stream-http "^2.7.2" - string_decoder "^1.0.0" - timers-browserify "^2.0.4" - tty-browserify "0.0.0" - url "^0.11.0" - util "^0.11.0" - vm-browserify "^1.0.1" - -node-releases@^1.1.70: - version "1.1.71" - resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-1.1.71.tgz#cb1334b179896b1c89ecfdd4b725fb7bbdfc7dbb" - integrity sha512-zR6HoT6LrLCRBwukmrVbHv0EpEQjksO6GmFcZQQuCAy139BEsoVKPYnf3jongYW83fAa1torLGYwxxky/p28sg== - -normalize-package-data@^2.4.0: - version "2.5.0" - resolved "https://registry.yarnpkg.com/normalize-package-data/-/normalize-package-data-2.5.0.tgz#e66db1838b200c1dfc233225d12cb36520e234a8" - integrity sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA== - dependencies: - hosted-git-info "^2.1.4" - resolve "^1.10.0" - semver "2 || 3 || 4 || 5" - validate-npm-package-license "^3.0.1" - -normalize-path@^2.1.1: - version "2.1.1" - resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-2.1.1.tgz#1ab28b556e198363a8c1a6f7e6fa20137fe6aed9" - integrity sha1-GrKLVW4Zg2Oowab35vogE3/mrtk= - dependencies: - remove-trailing-separator "^1.0.1" - -normalize-path@^3.0.0, normalize-path@~3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-3.0.0.tgz#0dcd69ff23a1c9b11fd0978316644a0388216a65" - integrity sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA== - -normalize-range@^0.1.2: - version "0.1.2" - resolved "https://registry.yarnpkg.com/normalize-range/-/normalize-range-0.1.2.tgz#2d10c06bdfd312ea9777695a4d28439456b75942" - integrity sha1-LRDAa9/TEuqXd2laTShDlFa3WUI= - -normalize-url@^3.0.0: - version "3.3.0" - resolved "https://registry.yarnpkg.com/normalize-url/-/normalize-url-3.3.0.tgz#b2e1c4dc4f7c6d57743df733a4f5978d18650559" - integrity sha512-U+JJi7duF1o+u2pynbp2zXDW2/PADgC30f0GsHZtRh+HOcXHnw137TrNlyxxRvWW5fjKd3bcLHPxofWuCjaeZg== - -normalize-url@^4.1.0: - version "4.5.0" - resolved "https://registry.yarnpkg.com/normalize-url/-/normalize-url-4.5.0.tgz#453354087e6ca96957bd8f5baf753f5982142129" - integrity sha512-2s47yzUxdexf1OhyRi4Em83iQk0aPvwTddtFz4hnSSw9dCEsLEGf6SwIO8ss/19S9iBb5sJaOuTvTGDeZI00BQ== - -npm-bundled@^1.0.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/npm-bundled/-/npm-bundled-1.1.1.tgz#1edd570865a94cdb1bc8220775e29466c9fb234b" - integrity sha512-gqkfgGePhTpAEgUsGEgcq1rqPXA+tv/aVBlgEzfXwA1yiUJF7xtEt3CtVwOjNYQOVknDk0F20w58Fnm3EtG0fA== - dependencies: - npm-normalize-package-bin "^1.0.1" - -npm-conf@^1.1.3: - version "1.1.3" - resolved "https://registry.yarnpkg.com/npm-conf/-/npm-conf-1.1.3.tgz#256cc47bd0e218c259c4e9550bf413bc2192aff9" - integrity sha512-Yic4bZHJOt9RCFbRP3GgpqhScOY4HH3V2P8yBj6CeYq118Qr+BLXqT2JvpJ00mryLESpgOxf5XlFv4ZjXxLScw== - dependencies: - config-chain "^1.1.11" - pify "^3.0.0" - -npm-install-checks@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/npm-install-checks/-/npm-install-checks-4.0.0.tgz#a37facc763a2fde0497ef2c6d0ac7c3fbe00d7b4" - integrity sha512-09OmyDkNLYwqKPOnbI8exiOZU2GVVmQp7tgez2BPi5OZC8M82elDAps7sxC4l//uSUtotWqoEIDwjRvWH4qz8w== - dependencies: - semver "^7.1.1" - -npm-normalize-package-bin@^1.0.0, npm-normalize-package-bin@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/npm-normalize-package-bin/-/npm-normalize-package-bin-1.0.1.tgz#6e79a41f23fd235c0623218228da7d9c23b8f6e2" - integrity sha512-EPfafl6JL5/rU+ot6P3gRSCpPDW5VmIzX959Ob1+ySFUuuYHWHekXpwdUZcKP5C+DS4GEtdJluwBjnsNDl+fSA== - -npm-package-arg@8.1.0: - version "8.1.0" - resolved "https://registry.yarnpkg.com/npm-package-arg/-/npm-package-arg-8.1.0.tgz#b5f6319418c3246a1c38e1a8fbaa06231bc5308f" - integrity sha512-/ep6QDxBkm9HvOhOg0heitSd7JHA1U7y1qhhlRlteYYAi9Pdb/ZV7FW5aHpkrpM8+P+4p/jjR8zCyKPBMBjSig== - dependencies: - hosted-git-info "^3.0.6" - semver "^7.0.0" - validate-npm-package-name "^3.0.0" - -npm-package-arg@^6.0.0, npm-package-arg@^6.1.0: - version "6.1.1" - resolved "https://registry.yarnpkg.com/npm-package-arg/-/npm-package-arg-6.1.1.tgz#02168cb0a49a2b75bf988a28698de7b529df5cb7" - integrity sha512-qBpssaL3IOZWi5vEKUKW0cO7kzLeT+EQO9W8RsLOZf76KF9E/K9+wH0C7t06HXPpaH8WH5xF1MExLuCwbTqRUg== - dependencies: - hosted-git-info "^2.7.1" - osenv "^0.1.5" - semver "^5.6.0" - validate-npm-package-name "^3.0.0" - -npm-package-arg@^8.0.0: - version "8.1.1" - resolved "https://registry.yarnpkg.com/npm-package-arg/-/npm-package-arg-8.1.1.tgz#00ebf16ac395c63318e67ce66780a06db6df1b04" - integrity sha512-CsP95FhWQDwNqiYS+Q0mZ7FAEDytDZAkNxQqea6IaAFJTAY9Lhhqyl0irU/6PMc7BGfUmnsbHcqxJD7XuVM/rg== - dependencies: - hosted-git-info "^3.0.6" - semver "^7.0.0" - validate-npm-package-name "^3.0.0" - -npm-packlist@^1.1.12: - version "1.4.8" - resolved "https://registry.yarnpkg.com/npm-packlist/-/npm-packlist-1.4.8.tgz#56ee6cc135b9f98ad3d51c1c95da22bbb9b2ef3e" - integrity sha512-5+AZgwru5IevF5ZdnFglB5wNlHG1AOOuw28WhUq8/8emhBmLv6jX5by4WJCh7lW0uSYZYS6DXqIsyZVIXRZU9A== - dependencies: - ignore-walk "^3.0.1" - npm-bundled "^1.0.1" - npm-normalize-package-bin "^1.0.1" - -npm-pick-manifest@6.1.0: - version "6.1.0" - resolved "https://registry.yarnpkg.com/npm-pick-manifest/-/npm-pick-manifest-6.1.0.tgz#2befed87b0fce956790f62d32afb56d7539c022a" - integrity sha512-ygs4k6f54ZxJXrzT0x34NybRlLeZ4+6nECAIbr2i0foTnijtS1TJiyzpqtuUAJOps/hO0tNDr8fRV5g+BtRlTw== - dependencies: - npm-install-checks "^4.0.0" - npm-package-arg "^8.0.0" - semver "^7.0.0" - -npm-pick-manifest@^3.0.0: - version "3.0.2" - resolved "https://registry.yarnpkg.com/npm-pick-manifest/-/npm-pick-manifest-3.0.2.tgz#f4d9e5fd4be2153e5f4e5f9b7be8dc419a99abb7" - integrity sha512-wNprTNg+X5nf+tDi+hbjdHhM4bX+mKqv6XmPh7B5eG+QY9VARfQPfCEH013H5GqfNj6ee8Ij2fg8yk0mzps1Vw== - dependencies: - figgy-pudding "^3.5.1" - npm-package-arg "^6.0.0" - semver "^5.4.1" - -npm-registry-fetch@^4.0.0: - version "4.0.7" - resolved "https://registry.yarnpkg.com/npm-registry-fetch/-/npm-registry-fetch-4.0.7.tgz#57951bf6541e0246b34c9f9a38ab73607c9449d7" - integrity sha512-cny9v0+Mq6Tjz+e0erFAB+RYJ/AVGzkjnISiobqP8OWj9c9FLoZZu8/SPSKJWE17F1tk4018wfjV+ZbIbqC7fQ== - dependencies: - JSONStream "^1.3.4" - bluebird "^3.5.1" - figgy-pudding "^3.4.1" - lru-cache "^5.1.1" - make-fetch-happen "^5.0.0" - npm-package-arg "^6.1.0" - safe-buffer "^5.2.0" - -npm-run-path@^2.0.0: - version "2.0.2" - resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-2.0.2.tgz#35a9232dfa35d7067b4cb2ddf2357b1871536c5f" - integrity sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8= - dependencies: - path-key "^2.0.0" - -nth-check@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/nth-check/-/nth-check-1.0.2.tgz#b2bd295c37e3dd58a3bf0700376663ba4d9cf05c" - integrity sha512-WeBOdju8SnzPN5vTUJYxYUxLeXpCaVP5i5e0LF8fg7WORF2Wd7wFX/pk0tYZk7s8T+J7VLy0Da6J1+wCT0AtHg== - dependencies: - boolbase "~1.0.0" - -num2fraction@^1.2.2: - version "1.2.2" - resolved "https://registry.yarnpkg.com/num2fraction/-/num2fraction-1.2.2.tgz#6f682b6a027a4e9ddfa4564cd2589d1d4e669ede" - integrity sha1-b2gragJ6Tp3fpFZM0lidHU5mnt4= - -oauth-sign@~0.9.0: - version "0.9.0" - resolved "https://registry.yarnpkg.com/oauth-sign/-/oauth-sign-0.9.0.tgz#47a7b016baa68b5fa0ecf3dee08a85c679ac6455" - integrity sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ== - -object-assign@^4.0.1, object-assign@^4.1.1: - version "4.1.1" - resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863" - integrity sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM= - -object-copy@^0.1.0: - version "0.1.0" - resolved "https://registry.yarnpkg.com/object-copy/-/object-copy-0.1.0.tgz#7e7d858b781bd7c991a41ba975ed3812754e998c" - integrity sha1-fn2Fi3gb18mRpBupde04EnVOmYw= - dependencies: - copy-descriptor "^0.1.0" - define-property "^0.2.5" - kind-of "^3.0.3" - -object-inspect@^1.9.0: - version "1.9.0" - resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.9.0.tgz#c90521d74e1127b67266ded3394ad6116986533a" - integrity sha512-i3Bp9iTqwhaLZBxGkRfo5ZbE07BQRT7MGu8+nNgwW9ItGp1TzCTw2DLEoWwjClxBjOFI/hWljTAmYGCEwmtnOw== - -object-is@^1.0.1: - version "1.1.5" - resolved "https://registry.yarnpkg.com/object-is/-/object-is-1.1.5.tgz#b9deeaa5fc7f1846a0faecdceec138e5778f53ac" - integrity sha512-3cyDsyHgtmi7I7DfSSI2LDp6SK2lwvtbg0p0R1e0RvTqF5ceGx+K2dfSjm1bKDMVCFEDAQvy+o8c6a7VujOddw== - dependencies: - call-bind "^1.0.2" - define-properties "^1.1.3" - -object-keys@^1.0.12, object-keys@^1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.1.1.tgz#1c47f272df277f3b1daf061677d9c82e2322c60e" - integrity sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA== - -object-visit@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/object-visit/-/object-visit-1.0.1.tgz#f79c4493af0c5377b59fe39d395e41042dd045bb" - integrity sha1-95xEk68MU3e1n+OdOV5BBC3QRbs= - dependencies: - isobject "^3.0.0" - -object.assign@^4.1.0, object.assign@^4.1.2: - version "4.1.2" - resolved "https://registry.yarnpkg.com/object.assign/-/object.assign-4.1.2.tgz#0ed54a342eceb37b38ff76eb831a0e788cb63940" - integrity sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ== - dependencies: - call-bind "^1.0.0" - define-properties "^1.1.3" - has-symbols "^1.0.1" - object-keys "^1.1.1" - -object.getownpropertydescriptors@^2.1.0: - version "2.1.2" - resolved "https://registry.yarnpkg.com/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.2.tgz#1bd63aeacf0d5d2d2f31b5e393b03a7c601a23f7" - integrity sha512-WtxeKSzfBjlzL+F9b7M7hewDzMwy+C8NRssHd1YrNlzHzIDrXcXiNOMrezdAEM4UXixgV+vvnyBeN7Rygl2ttQ== - dependencies: - call-bind "^1.0.2" - define-properties "^1.1.3" - es-abstract "^1.18.0-next.2" - -object.pick@^1.3.0: - version "1.3.0" - resolved "https://registry.yarnpkg.com/object.pick/-/object.pick-1.3.0.tgz#87a10ac4c1694bd2e1cbf53591a66141fb5dd747" - integrity sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c= - dependencies: - isobject "^3.0.1" - -object.values@^1.1.0: - version "1.1.3" - resolved "https://registry.yarnpkg.com/object.values/-/object.values-1.1.3.tgz#eaa8b1e17589f02f698db093f7c62ee1699742ee" - integrity sha512-nkF6PfDB9alkOUxpf1HNm/QlkeW3SReqL5WXeBLpEJJnlPSvRaDQpW3gQTksTN3fgJX4hL42RzKyOin6ff3tyw== - dependencies: - call-bind "^1.0.2" - define-properties "^1.1.3" - es-abstract "^1.18.0-next.2" - has "^1.0.3" - -obuf@^1.0.0, obuf@^1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/obuf/-/obuf-1.1.2.tgz#09bea3343d41859ebd446292d11c9d4db619084e" - integrity sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg== - -on-finished@~2.3.0: - version "2.3.0" - resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.3.0.tgz#20f1336481b083cd75337992a16971aa2d906947" - integrity sha1-IPEzZIGwg811M3mSoWlxqi2QaUc= - dependencies: - ee-first "1.1.1" - -on-headers@~1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/on-headers/-/on-headers-1.0.2.tgz#772b0ae6aaa525c399e489adfad90c403eb3c28f" - integrity sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA== - -once@^1.3.0, once@^1.3.1, once@^1.4.0: - version "1.4.0" - resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" - integrity sha1-WDsap3WWHUsROsF9nFC6753Xa9E= - dependencies: - wrappy "1" - -onetime@^5.1.0: - version "5.1.2" - resolved "https://registry.yarnpkg.com/onetime/-/onetime-5.1.2.tgz#d0e96ebb56b07476df1dd9c4806e5237985ca45e" - integrity sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg== - dependencies: - mimic-fn "^2.1.0" - -open@7.3.0: - version "7.3.0" - resolved "https://registry.yarnpkg.com/open/-/open-7.3.0.tgz#45461fdee46444f3645b6e14eb3ca94b82e1be69" - integrity sha512-mgLwQIx2F/ye9SmbrUkurZCnkoXyXyu9EbHtJZrICjVAJfyMArdHp3KkixGdZx1ZHFPNIwl0DDM1dFFqXbTLZw== - dependencies: - is-docker "^2.0.0" - is-wsl "^2.1.1" - -opn@^5.5.0: - version "5.5.0" - resolved "https://registry.yarnpkg.com/opn/-/opn-5.5.0.tgz#fc7164fab56d235904c51c3b27da6758ca3b9bfc" - integrity sha512-PqHpggC9bLV0VeWcdKhkpxY+3JTzetLSqTCWL/z/tFIbI6G8JCjondXklT1JinczLz2Xib62sSp0T/gKT4KksA== - dependencies: - is-wsl "^1.1.0" - -ora@5.1.0: - version "5.1.0" - resolved "https://registry.yarnpkg.com/ora/-/ora-5.1.0.tgz#b188cf8cd2d4d9b13fd25383bc3e5cba352c94f8" - integrity sha512-9tXIMPvjZ7hPTbk8DFq1f7Kow/HU/pQYB60JbNq+QnGwcyhWVZaQ4hM9zQDEsPxw/muLpgiHSaumUZxCAmod/w== - dependencies: - chalk "^4.1.0" - cli-cursor "^3.1.0" - cli-spinners "^2.4.0" - is-interactive "^1.0.0" - log-symbols "^4.0.0" - mute-stream "0.0.8" - strip-ansi "^6.0.0" - wcwidth "^1.0.1" - -original@^1.0.0: - version "1.0.2" - resolved "https://registry.yarnpkg.com/original/-/original-1.0.2.tgz#e442a61cffe1c5fd20a65f3261c26663b303f25f" - integrity sha512-hyBVl6iqqUOJ8FqRe+l/gS8H+kKYjrEndd5Pm1MfBtsEKA038HkkdbAl/72EAXGyonD/PFsvmVG+EvcIpliMBg== - dependencies: - url-parse "^1.4.3" - -os-browserify@^0.3.0: - version "0.3.0" - resolved "https://registry.yarnpkg.com/os-browserify/-/os-browserify-0.3.0.tgz#854373c7f5c2315914fc9bfc6bd8238fdda1ec27" - integrity sha1-hUNzx/XCMVkU/Jv8a9gjj92h7Cc= - -os-homedir@^1.0.0: - version "1.0.2" - resolved "https://registry.yarnpkg.com/os-homedir/-/os-homedir-1.0.2.tgz#ffbc4988336e0e833de0c168c7ef152121aa7fb3" - integrity sha1-/7xJiDNuDoM94MFox+8VISGqf7M= - -os-tmpdir@^1.0.0, os-tmpdir@~1.0.1, os-tmpdir@~1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/os-tmpdir/-/os-tmpdir-1.0.2.tgz#bbe67406c79aa85c5cfec766fe5734555dfa1274" - integrity sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ= - -osenv@^0.1.5: - version "0.1.5" - resolved "https://registry.yarnpkg.com/osenv/-/osenv-0.1.5.tgz#85cdfafaeb28e8677f416e287592b5f3f49ea410" - integrity sha512-0CWcCECdMVc2Rw3U5w9ZjqX6ga6ubk1xDVKxtBQPK7wis/0F2r9T6k4ydGYhecl7YUBxBVxhL5oisPsNxAPe2g== - dependencies: - os-homedir "^1.0.0" - os-tmpdir "^1.0.0" - -p-cancelable@^1.0.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/p-cancelable/-/p-cancelable-1.1.0.tgz#d078d15a3af409220c886f1d9a0ca2e441ab26cc" - integrity sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw== - -p-finally@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/p-finally/-/p-finally-1.0.0.tgz#3fbcfb15b899a44123b34b6dcc18b724336a2cae" - integrity sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4= - -p-limit@^2.0.0, p-limit@^2.2.0: - version "2.3.0" - resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-2.3.0.tgz#3dd33c647a214fdfffd835933eb086da0dc21db1" - integrity sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w== - dependencies: - p-try "^2.0.0" - -p-limit@^3.0.2: - version "3.1.0" - resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-3.1.0.tgz#e1daccbe78d0d1388ca18c64fea38e3e57e3706b" - integrity sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ== - dependencies: - yocto-queue "^0.1.0" - -p-locate@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-3.0.0.tgz#322d69a05c0264b25997d9f40cd8a891ab0064a4" - integrity sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ== - dependencies: - p-limit "^2.0.0" - -p-locate@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-4.1.0.tgz#a3428bb7088b3a60292f66919278b7c297ad4f07" - integrity sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A== - dependencies: - p-limit "^2.2.0" - -p-map@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/p-map/-/p-map-2.1.0.tgz#310928feef9c9ecc65b68b17693018a665cea175" - integrity sha512-y3b8Kpd8OAN444hxfBbFfj1FY/RjtTd8tzYwhUqNYXx0fXx2iX4maP4Qr6qhIKbQXI02wTLAda4fYUbDagTUFw== - -p-map@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/p-map/-/p-map-4.0.0.tgz#bb2f95a5eda2ec168ec9274e06a747c3e2904d2b" - integrity sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ== - dependencies: - aggregate-error "^3.0.0" - -p-retry@^3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/p-retry/-/p-retry-3.0.1.tgz#316b4c8893e2c8dc1cfa891f406c4b422bebf328" - integrity sha512-XE6G4+YTTkT2a0UWb2kjZe8xNwf8bIbnqpc/IS/idOBVhyves0mK5OJgeocjx7q5pvX/6m23xuzVPYT1uGM73w== - dependencies: - retry "^0.12.0" - -p-try@^2.0.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/p-try/-/p-try-2.2.0.tgz#cb2868540e313d61de58fafbe35ce9004d5540e6" - integrity sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ== - -pacote@9.5.12: - version "9.5.12" - resolved "https://registry.yarnpkg.com/pacote/-/pacote-9.5.12.tgz#1e11dd7a8d736bcc36b375a9804d41bb0377bf66" - integrity sha512-BUIj/4kKbwWg4RtnBncXPJd15piFSVNpTzY0rysSr3VnMowTYgkGKcaHrbReepAkjTr8lH2CVWRi58Spg2CicQ== - dependencies: - bluebird "^3.5.3" - cacache "^12.0.2" - chownr "^1.1.2" - figgy-pudding "^3.5.1" - get-stream "^4.1.0" - glob "^7.1.3" - infer-owner "^1.0.4" - lru-cache "^5.1.1" - make-fetch-happen "^5.0.0" - minimatch "^3.0.4" - minipass "^2.3.5" - mississippi "^3.0.0" - mkdirp "^0.5.1" - normalize-package-data "^2.4.0" - npm-normalize-package-bin "^1.0.0" - npm-package-arg "^6.1.0" - npm-packlist "^1.1.12" - npm-pick-manifest "^3.0.0" - npm-registry-fetch "^4.0.0" - osenv "^0.1.5" - promise-inflight "^1.0.1" - promise-retry "^1.1.1" - protoduck "^5.0.1" - rimraf "^2.6.2" - safe-buffer "^5.1.2" - semver "^5.6.0" - ssri "^6.0.1" - tar "^4.4.10" - unique-filename "^1.1.1" - which "^1.3.1" - -pako@~1.0.2, pako@~1.0.5: - version "1.0.11" - resolved "https://registry.yarnpkg.com/pako/-/pako-1.0.11.tgz#6c9599d340d54dfd3946380252a35705a6b992bf" - integrity sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw== - -parallel-transform@^1.1.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/parallel-transform/-/parallel-transform-1.2.0.tgz#9049ca37d6cb2182c3b1d2c720be94d14a5814fc" - integrity sha512-P2vSmIu38uIlvdcU7fDkyrxj33gTUy/ABO5ZUbGowxNCopBq/OoD42bP4UmMrJoPyk4Uqf0mu3mtWBhHCZD8yg== - dependencies: - cyclist "^1.0.1" - inherits "^2.0.3" - readable-stream "^2.1.5" - -parent-module@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/parent-module/-/parent-module-1.0.1.tgz#691d2709e78c79fae3a156622452d00762caaaa2" - integrity sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g== - dependencies: - callsites "^3.0.0" - -parse-asn1@^5.0.0, parse-asn1@^5.1.5: - version "5.1.6" - resolved "https://registry.yarnpkg.com/parse-asn1/-/parse-asn1-5.1.6.tgz#385080a3ec13cb62a62d39409cb3e88844cdaed4" - integrity sha512-RnZRo1EPU6JBnra2vGHj0yhp6ebyjBZpmUCLHWiFhxlzvBCCpAuZ7elsBp1PVAbQN0/04VD/19rfzlBSwLstMw== - dependencies: - asn1.js "^5.2.0" - browserify-aes "^1.0.0" - evp_bytestokey "^1.0.0" - pbkdf2 "^3.0.3" - safe-buffer "^5.1.1" - -parse-json@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-4.0.0.tgz#be35f5425be1f7f6c747184f98a788cb99477ee0" - integrity sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA= - dependencies: - error-ex "^1.3.1" - json-parse-better-errors "^1.0.1" - -parse-json@^5.0.0: - version "5.2.0" - resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-5.2.0.tgz#c76fc66dee54231c962b22bcc8a72cf2f99753cd" - integrity sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg== - dependencies: - "@babel/code-frame" "^7.0.0" - error-ex "^1.3.1" - json-parse-even-better-errors "^2.3.0" - lines-and-columns "^1.1.6" - -parse5-html-rewriting-stream@6.0.1: - version "6.0.1" - resolved "https://registry.yarnpkg.com/parse5-html-rewriting-stream/-/parse5-html-rewriting-stream-6.0.1.tgz#de1820559317ab4e451ea72dba05fddfd914480b" - integrity sha512-vwLQzynJVEfUlURxgnf51yAJDQTtVpNyGD8tKi2Za7m+akukNHxCcUQMAa/mUGLhCeicFdpy7Tlvj8ZNKadprg== - dependencies: - parse5 "^6.0.1" - parse5-sax-parser "^6.0.1" - -parse5-sax-parser@^6.0.1: - version "6.0.1" - resolved "https://registry.yarnpkg.com/parse5-sax-parser/-/parse5-sax-parser-6.0.1.tgz#98b4d366b5b266a7cd90b4b58906667af882daba" - integrity sha512-kXX+5S81lgESA0LsDuGjAlBybImAChYRMT+/uKCEXFBFOeEhS52qUCydGhU3qLRD8D9DVjaUo821WK7DM4iCeg== - dependencies: - parse5 "^6.0.1" - -parse5@^5.0.0: - version "5.1.1" - resolved "https://registry.yarnpkg.com/parse5/-/parse5-5.1.1.tgz#f68e4e5ba1852ac2cadc00f4555fff6c2abb6178" - integrity sha512-ugq4DFI0Ptb+WWjAdOK16+u/nHfiIrcE+sh8kZMaM0WllQKLI9rOUq6c2b7cwPkXdzfQESqvoqK6ug7U/Yyzug== - -parse5@^6.0.1: - version "6.0.1" - resolved "https://registry.yarnpkg.com/parse5/-/parse5-6.0.1.tgz#e1a1c085c569b3dc08321184f19a39cc27f7c30b" - integrity sha512-Ofn/CTFzRGTTxwpNEs9PP93gXShHcTq255nzRYSKe8AkVpZY7e1fpmTfOyoIvjP5HG7Z2ZM7VS9PPhQGW2pOpw== - -parseqs@0.0.6: - version "0.0.6" - resolved "https://registry.yarnpkg.com/parseqs/-/parseqs-0.0.6.tgz#8e4bb5a19d1cdc844a08ac974d34e273afa670d5" - integrity sha512-jeAGzMDbfSHHA091hr0r31eYfTig+29g3GKKE/PPbEQ65X0lmMwlEoqmhzu0iztID5uJpZsFlUPDP8ThPL7M8w== - -parseuri@0.0.6: - version "0.0.6" - resolved "https://registry.yarnpkg.com/parseuri/-/parseuri-0.0.6.tgz#e1496e829e3ac2ff47f39a4dd044b32823c4a25a" - integrity sha512-AUjen8sAkGgao7UyCX6Ahv0gIK2fABKmYjvP4xmy5JaKvcbTRueIqIPHLAfq30xJddqSE033IOMUSOMCcK3Sow== - -parseurl@~1.3.2, parseurl@~1.3.3: - version "1.3.3" - resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.3.tgz#9da19e7bee8d12dff0513ed5b76957793bc2e8d4" - integrity sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ== - -pascalcase@^0.1.1: - version "0.1.1" - resolved "https://registry.yarnpkg.com/pascalcase/-/pascalcase-0.1.1.tgz#b363e55e8006ca6fe21784d2db22bd15d7917f14" - integrity sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ= - -path-browserify@0.0.1: - version "0.0.1" - resolved "https://registry.yarnpkg.com/path-browserify/-/path-browserify-0.0.1.tgz#e6c4ddd7ed3aa27c68a20cc4e50e1a4ee83bbc4a" - integrity sha512-BapA40NHICOS+USX9SN4tyhq+A2RrN/Ws5F0Z5aMHDp98Fl86lX8Oti8B7uN93L4Ifv4fHOEA+pQw87gmMO/lQ== - -path-dirname@^1.0.0: - version "1.0.2" - resolved "https://registry.yarnpkg.com/path-dirname/-/path-dirname-1.0.2.tgz#cc33d24d525e099a5388c0336c6e32b9160609e0" - integrity sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA= - -path-exists@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-3.0.0.tgz#ce0ebeaa5f78cb18925ea7d810d7b59b010fd515" - integrity sha1-zg6+ql94yxiSXqfYENe1mwEP1RU= - -path-exists@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-4.0.0.tgz#513bdbe2d3b95d7762e8c1137efa195c6c61b5b3" - integrity sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w== - -path-is-absolute@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f" - integrity sha1-F0uSaHNVNP+8es5r9TpanhtcX18= - -path-is-inside@^1.0.1, path-is-inside@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/path-is-inside/-/path-is-inside-1.0.2.tgz#365417dede44430d1c11af61027facf074bdfc53" - integrity sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM= - -path-key@^2.0.0, path-key@^2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/path-key/-/path-key-2.0.1.tgz#411cadb574c5a140d3a4b1910d40d80cc9f40b40" - integrity sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A= - -path-parse@^1.0.6: - version "1.0.6" - resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.6.tgz#d62dbb5679405d72c4737ec58600e9ddcf06d24c" - integrity sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw== - -path-to-regexp@0.1.7: - version "0.1.7" - resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-0.1.7.tgz#df604178005f522f15eb4490e7247a1bfaa67f8c" - integrity sha1-32BBeABfUi8V60SQ5yR6G/qmf4w= - -path-type@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/path-type/-/path-type-4.0.0.tgz#84ed01c0a7ba380afe09d90a8c180dcd9d03043b" - integrity sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw== - -pbkdf2@^3.0.3: - version "3.1.1" - resolved "https://registry.yarnpkg.com/pbkdf2/-/pbkdf2-3.1.1.tgz#cb8724b0fada984596856d1a6ebafd3584654b94" - integrity sha512-4Ejy1OPxi9f2tt1rRV7Go7zmfDQ+ZectEQz3VGUQhgq62HtIRPDyG/JtnwIxs6x3uNMwo2V7q1fMvKjb+Tnpqg== - dependencies: - create-hash "^1.1.2" - create-hmac "^1.1.4" - ripemd160 "^2.0.1" - safe-buffer "^5.0.1" - sha.js "^2.4.8" - -pend@~1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/pend/-/pend-1.2.0.tgz#7a57eb550a6783f9115331fcf4663d5c8e007a50" - integrity sha1-elfrVQpng/kRUzH89GY9XI4AelA= - -performance-now@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/performance-now/-/performance-now-2.1.0.tgz#6309f4e0e5fa913ec1c69307ae364b4b377c9e7b" - integrity sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns= - -picomatch@^2.0.4, picomatch@^2.0.5, picomatch@^2.2.1: - version "2.2.2" - resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.2.2.tgz#21f333e9b6b8eaff02468f5146ea406d345f4dad" - integrity sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg== - -pify@^2.0.0, pify@^2.3.0: - version "2.3.0" - resolved "https://registry.yarnpkg.com/pify/-/pify-2.3.0.tgz#ed141a6ac043a849ea588498e7dca8b15330e90c" - integrity sha1-7RQaasBDqEnqWISY59yosVMw6Qw= - -pify@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/pify/-/pify-3.0.0.tgz#e5a4acd2c101fdf3d9a4d07f0dbc4db49dd28176" - integrity sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY= - -pify@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/pify/-/pify-4.0.1.tgz#4b2cd25c50d598735c50292224fd8c6df41e3231" - integrity sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g== - -pinkie-promise@^2.0.0: - version "2.0.1" - resolved "https://registry.yarnpkg.com/pinkie-promise/-/pinkie-promise-2.0.1.tgz#2135d6dfa7a358c069ac9b178776288228450ffa" - integrity sha1-ITXW36ejWMBprJsXh3YogihFD/o= - dependencies: - pinkie "^2.0.0" - -pinkie@^2.0.0: - version "2.0.4" - resolved "https://registry.yarnpkg.com/pinkie/-/pinkie-2.0.4.tgz#72556b80cfa0d48a974e80e77248e80ed4f7f870" - integrity sha1-clVrgM+g1IqXToDnckjoDtT3+HA= - -pkg-dir@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/pkg-dir/-/pkg-dir-3.0.0.tgz#2749020f239ed990881b1f71210d51eb6523bea3" - integrity sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw== - dependencies: - find-up "^3.0.0" - -pkg-dir@^4.1.0: - version "4.2.0" - resolved "https://registry.yarnpkg.com/pkg-dir/-/pkg-dir-4.2.0.tgz#f099133df7ede422e81d1d8448270eeb3e4261f3" - integrity sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ== - dependencies: - find-up "^4.0.0" - -pnp-webpack-plugin@1.6.4: - version "1.6.4" - resolved "https://registry.yarnpkg.com/pnp-webpack-plugin/-/pnp-webpack-plugin-1.6.4.tgz#c9711ac4dc48a685dabafc86f8b6dd9f8df84149" - integrity sha512-7Wjy+9E3WwLOEL30D+m8TSTF7qJJUJLONBnwQp0518siuMxUQUbgZwssaFX+QKlZkjHZcw/IpZCt/H0srrntSg== - dependencies: - ts-pnp "^1.1.6" - -portfinder@^1.0.26: - version "1.0.28" - resolved "https://registry.yarnpkg.com/portfinder/-/portfinder-1.0.28.tgz#67c4622852bd5374dd1dd900f779f53462fac778" - integrity sha512-Se+2isanIcEqf2XMHjyUKskczxbPH7dQnlMjXX6+dybayyHvAf/TCgyMRlzf/B6QDhAEFOGes0pzRo3by4AbMA== - dependencies: - async "^2.6.2" - debug "^3.1.1" - mkdirp "^0.5.5" - -posix-character-classes@^0.1.0: - version "0.1.1" - resolved "https://registry.yarnpkg.com/posix-character-classes/-/posix-character-classes-0.1.1.tgz#01eac0fe3b5af71a2a6c02feabb8c1fef7e00eab" - integrity sha1-AerA/jta9xoqbAL+q7jB/vfgDqs= - -postcss-calc@^7.0.1: - version "7.0.5" - resolved "https://registry.yarnpkg.com/postcss-calc/-/postcss-calc-7.0.5.tgz#f8a6e99f12e619c2ebc23cf6c486fdc15860933e" - integrity sha512-1tKHutbGtLtEZF6PT4JSihCHfIVldU72mZ8SdZHIYriIZ9fh9k9aWSppaT8rHsyI3dX+KSR+W+Ix9BMY3AODrg== - dependencies: - postcss "^7.0.27" - postcss-selector-parser "^6.0.2" - postcss-value-parser "^4.0.2" - -postcss-colormin@^4.0.3: - version "4.0.3" - resolved "https://registry.yarnpkg.com/postcss-colormin/-/postcss-colormin-4.0.3.tgz#ae060bce93ed794ac71264f08132d550956bd381" - integrity sha512-WyQFAdDZpExQh32j0U0feWisZ0dmOtPl44qYmJKkq9xFWY3p+4qnRzCHeNrkeRhwPHz9bQ3mo0/yVkaply0MNw== - dependencies: - browserslist "^4.0.0" - color "^3.0.0" - has "^1.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-convert-values@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/postcss-convert-values/-/postcss-convert-values-4.0.1.tgz#ca3813ed4da0f812f9d43703584e449ebe189a7f" - integrity sha512-Kisdo1y77KUC0Jmn0OXU/COOJbzM8cImvw1ZFsBgBgMgb1iL23Zs/LXRe3r+EZqM3vGYKdQ2YJVQ5VkJI+zEJQ== - dependencies: - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-discard-comments@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-discard-comments/-/postcss-discard-comments-4.0.2.tgz#1fbabd2c246bff6aaad7997b2b0918f4d7af4033" - integrity sha512-RJutN259iuRf3IW7GZyLM5Sw4GLTOH8FmsXBnv8Ab/Tc2k4SR4qbV4DNbyyY4+Sjo362SyDmW2DQ7lBSChrpkg== - dependencies: - postcss "^7.0.0" - -postcss-discard-duplicates@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-discard-duplicates/-/postcss-discard-duplicates-4.0.2.tgz#3fe133cd3c82282e550fc9b239176a9207b784eb" - integrity sha512-ZNQfR1gPNAiXZhgENFfEglF93pciw0WxMkJeVmw8eF+JZBbMD7jp6C67GqJAXVZP2BWbOztKfbsdmMp/k8c6oQ== - dependencies: - postcss "^7.0.0" - -postcss-discard-empty@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/postcss-discard-empty/-/postcss-discard-empty-4.0.1.tgz#c8c951e9f73ed9428019458444a02ad90bb9f765" - integrity sha512-B9miTzbznhDjTfjvipfHoqbWKwd0Mj+/fL5s1QOz06wufguil+Xheo4XpOnc4NqKYBCNqqEzgPv2aPBIJLox0w== - dependencies: - postcss "^7.0.0" - -postcss-discard-overridden@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/postcss-discard-overridden/-/postcss-discard-overridden-4.0.1.tgz#652aef8a96726f029f5e3e00146ee7a4e755ff57" - integrity sha512-IYY2bEDD7g1XM1IDEsUT4//iEYCxAmP5oDSFMVU/JVvT7gh+l4fmjciLqGgwjdWpQIdb0Che2VX00QObS5+cTg== - dependencies: - postcss "^7.0.0" - -postcss-import@12.0.1: - version "12.0.1" - resolved "https://registry.yarnpkg.com/postcss-import/-/postcss-import-12.0.1.tgz#cf8c7ab0b5ccab5649024536e565f841928b7153" - integrity sha512-3Gti33dmCjyKBgimqGxL3vcV8w9+bsHwO5UrBawp796+jdardbcFl4RP5w/76BwNL7aGzpKstIfF9I+kdE8pTw== - dependencies: - postcss "^7.0.1" - postcss-value-parser "^3.2.3" - read-cache "^1.0.0" - resolve "^1.1.7" - -postcss-loader@4.0.4: - version "4.0.4" - resolved "https://registry.yarnpkg.com/postcss-loader/-/postcss-loader-4.0.4.tgz#b2d005b52e008a44991cf8123bee207e635eb53e" - integrity sha512-pntA9zIR14drQo84yGTjQJg1m7T0DkXR4vXYHBngiRZdJtEeCrojL6lOpqUanMzG375lIJbT4Yug85zC/AJWGw== - dependencies: - cosmiconfig "^7.0.0" - klona "^2.0.4" - loader-utils "^2.0.0" - schema-utils "^3.0.0" - semver "^7.3.2" - -postcss-merge-longhand@^4.0.11: - version "4.0.11" - resolved "https://registry.yarnpkg.com/postcss-merge-longhand/-/postcss-merge-longhand-4.0.11.tgz#62f49a13e4a0ee04e7b98f42bb16062ca2549e24" - integrity sha512-alx/zmoeXvJjp7L4mxEMjh8lxVlDFX1gqWHzaaQewwMZiVhLo42TEClKaeHbRf6J7j82ZOdTJ808RtN0ZOZwvw== - dependencies: - css-color-names "0.0.4" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - stylehacks "^4.0.0" - -postcss-merge-rules@^4.0.3: - version "4.0.3" - resolved "https://registry.yarnpkg.com/postcss-merge-rules/-/postcss-merge-rules-4.0.3.tgz#362bea4ff5a1f98e4075a713c6cb25aefef9a650" - integrity sha512-U7e3r1SbvYzO0Jr3UT/zKBVgYYyhAz0aitvGIYOYK5CPmkNih+WDSsS5tvPrJ8YMQYlEMvsZIiqmn7HdFUaeEQ== - dependencies: - browserslist "^4.0.0" - caniuse-api "^3.0.0" - cssnano-util-same-parent "^4.0.0" - postcss "^7.0.0" - postcss-selector-parser "^3.0.0" - vendors "^1.0.0" - -postcss-minify-font-values@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-minify-font-values/-/postcss-minify-font-values-4.0.2.tgz#cd4c344cce474343fac5d82206ab2cbcb8afd5a6" - integrity sha512-j85oO6OnRU9zPf04+PZv1LYIYOprWm6IA6zkXkrJXyRveDEuQggG6tvoy8ir8ZwjLxLuGfNkCZEQG7zan+Hbtg== - dependencies: - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-minify-gradients@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-minify-gradients/-/postcss-minify-gradients-4.0.2.tgz#93b29c2ff5099c535eecda56c4aa6e665a663471" - integrity sha512-qKPfwlONdcf/AndP1U8SJ/uzIJtowHlMaSioKzebAXSG4iJthlWC9iSWznQcX4f66gIWX44RSA841HTHj3wK+Q== - dependencies: - cssnano-util-get-arguments "^4.0.0" - is-color-stop "^1.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-minify-params@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-minify-params/-/postcss-minify-params-4.0.2.tgz#6b9cef030c11e35261f95f618c90036d680db874" - integrity sha512-G7eWyzEx0xL4/wiBBJxJOz48zAKV2WG3iZOqVhPet/9geefm/Px5uo1fzlHu+DOjT+m0Mmiz3jkQzVHe6wxAWg== - dependencies: - alphanum-sort "^1.0.0" - browserslist "^4.0.0" - cssnano-util-get-arguments "^4.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - uniqs "^2.0.0" - -postcss-minify-selectors@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-minify-selectors/-/postcss-minify-selectors-4.0.2.tgz#e2e5eb40bfee500d0cd9243500f5f8ea4262fbd8" - integrity sha512-D5S1iViljXBj9kflQo4YutWnJmwm8VvIsU1GeXJGiG9j8CIg9zs4voPMdQDUmIxetUOh60VilsNzCiAFTOqu3g== - dependencies: - alphanum-sort "^1.0.0" - has "^1.0.0" - postcss "^7.0.0" - postcss-selector-parser "^3.0.0" - -postcss-modules-extract-imports@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/postcss-modules-extract-imports/-/postcss-modules-extract-imports-2.0.0.tgz#818719a1ae1da325f9832446b01136eeb493cd7e" - integrity sha512-LaYLDNS4SG8Q5WAWqIJgdHPJrDDr/Lv775rMBFUbgjTz6j34lUznACHcdRWroPvXANP2Vj7yNK57vp9eFqzLWQ== - dependencies: - postcss "^7.0.5" - -postcss-modules-local-by-default@^3.0.3: - version "3.0.3" - resolved "https://registry.yarnpkg.com/postcss-modules-local-by-default/-/postcss-modules-local-by-default-3.0.3.tgz#bb14e0cc78279d504dbdcbfd7e0ca28993ffbbb0" - integrity sha512-e3xDq+LotiGesympRlKNgaJ0PCzoUIdpH0dj47iWAui/kyTgh3CiAr1qP54uodmJhl6p9rN6BoNcdEDVJx9RDw== - dependencies: - icss-utils "^4.1.1" - postcss "^7.0.32" - postcss-selector-parser "^6.0.2" - postcss-value-parser "^4.1.0" - -postcss-modules-scope@^2.2.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/postcss-modules-scope/-/postcss-modules-scope-2.2.0.tgz#385cae013cc7743f5a7d7602d1073a89eaae62ee" - integrity sha512-YyEgsTMRpNd+HmyC7H/mh3y+MeFWevy7V1evVhJWewmMbjDHIbZbOXICC2y+m1xI1UVfIT1HMW/O04Hxyu9oXQ== - dependencies: - postcss "^7.0.6" - postcss-selector-parser "^6.0.0" - -postcss-modules-values@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/postcss-modules-values/-/postcss-modules-values-3.0.0.tgz#5b5000d6ebae29b4255301b4a3a54574423e7f10" - integrity sha512-1//E5jCBrZ9DmRX+zCtmQtRSV6PV42Ix7Bzj9GbwJceduuf7IqP8MgeTXuRDHOWj2m0VzZD5+roFWDuU8RQjcg== - dependencies: - icss-utils "^4.0.0" - postcss "^7.0.6" - -postcss-normalize-charset@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/postcss-normalize-charset/-/postcss-normalize-charset-4.0.1.tgz#8b35add3aee83a136b0471e0d59be58a50285dd4" - integrity sha512-gMXCrrlWh6G27U0hF3vNvR3w8I1s2wOBILvA87iNXaPvSNo5uZAMYsZG7XjCUf1eVxuPfyL4TJ7++SGZLc9A3g== - dependencies: - postcss "^7.0.0" - -postcss-normalize-display-values@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-normalize-display-values/-/postcss-normalize-display-values-4.0.2.tgz#0dbe04a4ce9063d4667ed2be476bb830c825935a" - integrity sha512-3F2jcsaMW7+VtRMAqf/3m4cPFhPD3EFRgNs18u+k3lTJJlVe7d0YPO+bnwqo2xg8YiRpDXJI2u8A0wqJxMsQuQ== - dependencies: - cssnano-util-get-match "^4.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-normalize-positions@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-normalize-positions/-/postcss-normalize-positions-4.0.2.tgz#05f757f84f260437378368a91f8932d4b102917f" - integrity sha512-Dlf3/9AxpxE+NF1fJxYDeggi5WwV35MXGFnnoccP/9qDtFrTArZ0D0R+iKcg5WsUd8nUYMIl8yXDCtcrT8JrdA== - dependencies: - cssnano-util-get-arguments "^4.0.0" - has "^1.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-normalize-repeat-style@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-normalize-repeat-style/-/postcss-normalize-repeat-style-4.0.2.tgz#c4ebbc289f3991a028d44751cbdd11918b17910c" - integrity sha512-qvigdYYMpSuoFs3Is/f5nHdRLJN/ITA7huIoCyqqENJe9PvPmLhNLMu7QTjPdtnVf6OcYYO5SHonx4+fbJE1+Q== - dependencies: - cssnano-util-get-arguments "^4.0.0" - cssnano-util-get-match "^4.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-normalize-string@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-normalize-string/-/postcss-normalize-string-4.0.2.tgz#cd44c40ab07a0c7a36dc5e99aace1eca4ec2690c" - integrity sha512-RrERod97Dnwqq49WNz8qo66ps0swYZDSb6rM57kN2J+aoyEAJfZ6bMx0sx/F9TIEX0xthPGCmeyiam/jXif0eA== - dependencies: - has "^1.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-normalize-timing-functions@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-normalize-timing-functions/-/postcss-normalize-timing-functions-4.0.2.tgz#8e009ca2a3949cdaf8ad23e6b6ab99cb5e7d28d9" - integrity sha512-acwJY95edP762e++00Ehq9L4sZCEcOPyaHwoaFOhIwWCDfik6YvqsYNxckee65JHLKzuNSSmAdxwD2Cud1Z54A== - dependencies: - cssnano-util-get-match "^4.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-normalize-unicode@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/postcss-normalize-unicode/-/postcss-normalize-unicode-4.0.1.tgz#841bd48fdcf3019ad4baa7493a3d363b52ae1cfb" - integrity sha512-od18Uq2wCYn+vZ/qCOeutvHjB5jm57ToxRaMeNuf0nWVHaP9Hua56QyMF6fs/4FSUnVIw0CBPsU0K4LnBPwYwg== - dependencies: - browserslist "^4.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-normalize-url@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/postcss-normalize-url/-/postcss-normalize-url-4.0.1.tgz#10e437f86bc7c7e58f7b9652ed878daaa95faae1" - integrity sha512-p5oVaF4+IHwu7VpMan/SSpmpYxcJMtkGppYf0VbdH5B6hN8YNmVyJLuY9FmLQTzY3fag5ESUUHDqM+heid0UVA== - dependencies: - is-absolute-url "^2.0.0" - normalize-url "^3.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-normalize-whitespace@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-normalize-whitespace/-/postcss-normalize-whitespace-4.0.2.tgz#bf1d4070fe4fcea87d1348e825d8cc0c5faa7d82" - integrity sha512-tO8QIgrsI3p95r8fyqKV+ufKlSHh9hMJqACqbv2XknufqEDhDvbguXGBBqxw9nsQoXWf0qOqppziKJKHMD4GtA== - dependencies: - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-ordered-values@^4.1.2: - version "4.1.2" - resolved "https://registry.yarnpkg.com/postcss-ordered-values/-/postcss-ordered-values-4.1.2.tgz#0cf75c820ec7d5c4d280189559e0b571ebac0eee" - integrity sha512-2fCObh5UanxvSxeXrtLtlwVThBvHn6MQcu4ksNT2tsaV2Fg76R2CV98W7wNSlX+5/pFwEyaDwKLLoEV7uRybAw== - dependencies: - cssnano-util-get-arguments "^4.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-reduce-initial@^4.0.3: - version "4.0.3" - resolved "https://registry.yarnpkg.com/postcss-reduce-initial/-/postcss-reduce-initial-4.0.3.tgz#7fd42ebea5e9c814609639e2c2e84ae270ba48df" - integrity sha512-gKWmR5aUulSjbzOfD9AlJiHCGH6AEVLaM0AV+aSioxUDd16qXP1PCh8d1/BGVvpdWn8k/HiK7n6TjeoXN1F7DA== - dependencies: - browserslist "^4.0.0" - caniuse-api "^3.0.0" - has "^1.0.0" - postcss "^7.0.0" - -postcss-reduce-transforms@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-reduce-transforms/-/postcss-reduce-transforms-4.0.2.tgz#17efa405eacc6e07be3414a5ca2d1074681d4e29" - integrity sha512-EEVig1Q2QJ4ELpJXMZR8Vt5DQx8/mo+dGWSR7vWXqcob2gQLyQGsionYcGKATXvQzMPn6DSN1vTN7yFximdIAg== - dependencies: - cssnano-util-get-match "^4.0.0" - has "^1.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - -postcss-selector-parser@^3.0.0: - version "3.1.2" - resolved "https://registry.yarnpkg.com/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz#b310f5c4c0fdaf76f94902bbaa30db6aa84f5270" - integrity sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA== - dependencies: - dot-prop "^5.2.0" - indexes-of "^1.0.1" - uniq "^1.0.1" - -postcss-selector-parser@^6.0.0, postcss-selector-parser@^6.0.2: - version "6.0.4" - resolved "https://registry.yarnpkg.com/postcss-selector-parser/-/postcss-selector-parser-6.0.4.tgz#56075a1380a04604c38b063ea7767a129af5c2b3" - integrity sha512-gjMeXBempyInaBqpp8gODmwZ52WaYsVOsfr4L4lDQ7n3ncD6mEyySiDtgzCT+NYC0mmeOLvtsF8iaEf0YT6dBw== - dependencies: - cssesc "^3.0.0" - indexes-of "^1.0.1" - uniq "^1.0.1" - util-deprecate "^1.0.2" - -postcss-svgo@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/postcss-svgo/-/postcss-svgo-4.0.2.tgz#17b997bc711b333bab143aaed3b8d3d6e3d38258" - integrity sha512-C6wyjo3VwFm0QgBy+Fu7gCYOkCmgmClghO+pjcxvrcBKtiKt0uCF+hvbMO1fyv5BMImRK90SMb+dwUnfbGd+jw== - dependencies: - is-svg "^3.0.0" - postcss "^7.0.0" - postcss-value-parser "^3.0.0" - svgo "^1.0.0" - -postcss-unique-selectors@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/postcss-unique-selectors/-/postcss-unique-selectors-4.0.1.tgz#9446911f3289bfd64c6d680f073c03b1f9ee4bac" - integrity sha512-+JanVaryLo9QwZjKrmJgkI4Fn8SBgRO6WXQBJi7KiAVPlmxikB5Jzc4EvXMT2H0/m0RjrVVm9rGNhZddm/8Spg== - dependencies: - alphanum-sort "^1.0.0" - postcss "^7.0.0" - uniqs "^2.0.0" - -postcss-value-parser@^3.0.0, postcss-value-parser@^3.2.3: - version "3.3.1" - resolved "https://registry.yarnpkg.com/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz#9ff822547e2893213cf1c30efa51ac5fd1ba8281" - integrity sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ== - -postcss-value-parser@^4.0.2, postcss-value-parser@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/postcss-value-parser/-/postcss-value-parser-4.1.0.tgz#443f6a20ced6481a2bda4fa8532a6e55d789a2cb" - integrity sha512-97DXOFbQJhk71ne5/Mt6cOu6yxsSfM0QGQyl0L25Gca4yGWEGJaig7l7gbCX623VqTBNGLRLaVUCnNkcedlRSQ== - -postcss@7.0.21: - version "7.0.21" - resolved "https://registry.yarnpkg.com/postcss/-/postcss-7.0.21.tgz#06bb07824c19c2021c5d056d5b10c35b989f7e17" - integrity sha512-uIFtJElxJo29QC753JzhidoAhvp/e/Exezkdhfmt8AymWT6/5B7W1WmponYWkHk2eg6sONyTch0A3nkMPun3SQ== - dependencies: - chalk "^2.4.2" - source-map "^0.6.1" - supports-color "^6.1.0" - -postcss@7.0.32: - version "7.0.32" - resolved "https://registry.yarnpkg.com/postcss/-/postcss-7.0.32.tgz#4310d6ee347053da3433db2be492883d62cec59d" - integrity sha512-03eXong5NLnNCD05xscnGKGDZ98CyzoqPSMjOe6SuoQY7Z2hIj0Ld1g/O/UQRuOle2aRtiIRDg9tDcTGAkLfKw== - dependencies: - chalk "^2.4.2" - source-map "^0.6.1" - supports-color "^6.1.0" - -postcss@^7.0.0, postcss@^7.0.1, postcss@^7.0.14, postcss@^7.0.27, postcss@^7.0.32, postcss@^7.0.5, postcss@^7.0.6: - version "7.0.35" - resolved "https://registry.yarnpkg.com/postcss/-/postcss-7.0.35.tgz#d2be00b998f7f211d8a276974079f2e92b970e24" - integrity sha512-3QT8bBJeX/S5zKTTjTCIjRF3If4avAT6kqxcASlTWEtAFCb9NH0OUxNDfgZSWdP5fJnBYCMEWkIFfWeugjzYMg== - dependencies: - chalk "^2.4.2" - source-map "^0.6.1" - supports-color "^6.1.0" - -prepend-http@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/prepend-http/-/prepend-http-2.0.0.tgz#e92434bfa5ea8c19f41cdfd401d741a3c819d897" - integrity sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc= - -process-nextick-args@~2.0.0: - version "2.0.1" - resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.1.tgz#7820d9b16120cc55ca9ae7792680ae7dba6d7fe2" - integrity sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag== - -process@^0.11.10: - version "0.11.10" - resolved "https://registry.yarnpkg.com/process/-/process-0.11.10.tgz#7332300e840161bda3e69a1d1d91a7d4bc16f182" - integrity sha1-czIwDoQBYb2j5podHZGn1LwW8YI= - -progress@^2.0.3: - version "2.0.3" - resolved "https://registry.yarnpkg.com/progress/-/progress-2.0.3.tgz#7e8cf8d8f5b8f239c1bc68beb4eb78567d572ef8" - integrity sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA== - -promise-inflight@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/promise-inflight/-/promise-inflight-1.0.1.tgz#98472870bf228132fcbdd868129bad12c3c029e3" - integrity sha1-mEcocL8igTL8vdhoEputEsPAKeM= - -promise-retry@^1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/promise-retry/-/promise-retry-1.1.1.tgz#6739e968e3051da20ce6497fb2b50f6911df3d6d" - integrity sha1-ZznpaOMFHaIM5kl/srUPaRHfPW0= - dependencies: - err-code "^1.0.0" - retry "^0.10.0" - -proto-list@~1.2.1: - version "1.2.4" - resolved "https://registry.yarnpkg.com/proto-list/-/proto-list-1.2.4.tgz#212d5bfe1318306a420f6402b8e26ff39647a849" - integrity sha1-IS1b/hMYMGpCD2QCuOJv85ZHqEk= - -protoduck@^5.0.1: - version "5.0.1" - resolved "https://registry.yarnpkg.com/protoduck/-/protoduck-5.0.1.tgz#03c3659ca18007b69a50fd82a7ebcc516261151f" - integrity sha512-WxoCeDCoCBY55BMvj4cAEjdVUFGRWed9ZxPlqTKYyw1nDDTQ4pqmnIMAGfJlg7Dx35uB/M+PHJPTmGOvaCaPTg== - dependencies: - genfun "^5.0.0" - -protractor@~7.0.0: - version "7.0.0" - resolved "https://registry.yarnpkg.com/protractor/-/protractor-7.0.0.tgz#c3e263608bd72e2c2dc802b11a772711a4792d03" - integrity sha512-UqkFjivi4GcvUQYzqGYNe0mLzfn5jiLmO8w9nMhQoJRLhy2grJonpga2IWhI6yJO30LibWXJJtA4MOIZD2GgZw== - dependencies: - "@types/q" "^0.0.32" - "@types/selenium-webdriver" "^3.0.0" - blocking-proxy "^1.0.0" - browserstack "^1.5.1" - chalk "^1.1.3" - glob "^7.0.3" - jasmine "2.8.0" - jasminewd2 "^2.1.0" - q "1.4.1" - saucelabs "^1.5.0" - selenium-webdriver "3.6.0" - source-map-support "~0.4.0" - webdriver-js-extender "2.1.0" - webdriver-manager "^12.1.7" - yargs "^15.3.1" - -proxy-addr@~2.0.5: - version "2.0.6" - resolved "https://registry.yarnpkg.com/proxy-addr/-/proxy-addr-2.0.6.tgz#fdc2336505447d3f2f2c638ed272caf614bbb2bf" - integrity sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw== - dependencies: - forwarded "~0.1.2" - ipaddr.js "1.9.1" - -prr@~1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/prr/-/prr-1.0.1.tgz#d3fc114ba06995a45ec6893f484ceb1d78f5f476" - integrity sha1-0/wRS6BplaRexok/SEzrHXj19HY= - -psl@^1.1.28: - version "1.8.0" - resolved "https://registry.yarnpkg.com/psl/-/psl-1.8.0.tgz#9326f8bcfb013adcc005fdff056acce020e51c24" - integrity sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ== - -public-encrypt@^4.0.0: - version "4.0.3" - resolved "https://registry.yarnpkg.com/public-encrypt/-/public-encrypt-4.0.3.tgz#4fcc9d77a07e48ba7527e7cbe0de33d0701331e0" - integrity sha512-zVpa8oKZSz5bTMTFClc1fQOnyyEzpl5ozpi1B5YcvBrdohMjH2rfsBtyXcuNuwjsDIXmBYlF2N5FlJYhR29t8Q== - dependencies: - bn.js "^4.1.0" - browserify-rsa "^4.0.0" - create-hash "^1.1.0" - parse-asn1 "^5.0.0" - randombytes "^2.0.1" - safe-buffer "^5.1.2" - -pump@^2.0.0: - version "2.0.1" - resolved "https://registry.yarnpkg.com/pump/-/pump-2.0.1.tgz#12399add6e4cf7526d973cbc8b5ce2e2908b3909" - integrity sha512-ruPMNRkN3MHP1cWJc9OWr+T/xDP0jhXYCLfJcBuX54hhfIBnaQmAUMfDcG4DM5UMWByBbJY69QSphm3jtDKIkA== - dependencies: - end-of-stream "^1.1.0" - once "^1.3.1" - -pump@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/pump/-/pump-3.0.0.tgz#b4a2116815bde2f4e1ea602354e8c75565107a64" - integrity sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww== - dependencies: - end-of-stream "^1.1.0" - once "^1.3.1" - -pumpify@^1.3.3: - version "1.5.1" - resolved "https://registry.yarnpkg.com/pumpify/-/pumpify-1.5.1.tgz#36513be246ab27570b1a374a5ce278bfd74370ce" - integrity sha512-oClZI37HvuUJJxSKKrC17bZ9Cu0ZYhEAGPsPUy9KlMUmv9dKX2o77RUmq7f3XjIxbwyGwYzbzQ1L2Ks8sIradQ== - dependencies: - duplexify "^3.6.0" - inherits "^2.0.3" - pump "^2.0.0" - -punycode@1.3.2: - version "1.3.2" - resolved "https://registry.yarnpkg.com/punycode/-/punycode-1.3.2.tgz#9653a036fb7c1ee42342f2325cceefea3926c48d" - integrity sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0= - -punycode@^1.2.4: - version "1.4.1" - resolved "https://registry.yarnpkg.com/punycode/-/punycode-1.4.1.tgz#c0d5a63b2718800ad8e1eb0fa5269c84dd41845e" - integrity sha1-wNWmOycYgArY4esPpSachN1BhF4= - -punycode@^2.1.0, punycode@^2.1.1: - version "2.1.1" - resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.1.1.tgz#b58b010ac40c22c5657616c8d2c2c02c7bf479ec" - integrity sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A== - -q@1.4.1: - version "1.4.1" - resolved "https://registry.yarnpkg.com/q/-/q-1.4.1.tgz#55705bcd93c5f3673530c2c2cbc0c2b3addc286e" - integrity sha1-VXBbzZPF82c1MMLCy8DCs63cKG4= - -q@^1.1.2, q@^1.4.1: - version "1.5.1" - resolved "https://registry.yarnpkg.com/q/-/q-1.5.1.tgz#7e32f75b41381291d04611f1bf14109ac00651d7" - integrity sha1-fjL3W0E4EpHQRhHxvxQQmsAGUdc= - -qjobs@^1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/qjobs/-/qjobs-1.2.0.tgz#c45e9c61800bd087ef88d7e256423bdd49e5d071" - integrity sha512-8YOJEHtxpySA3fFDyCRxA+UUV+fA+rTWnuWvylOK/NCjhY+b4ocCtmu8TtsWb+mYeU+GCHf/S66KZF/AsteKHg== - -qs@6.7.0: - version "6.7.0" - resolved "https://registry.yarnpkg.com/qs/-/qs-6.7.0.tgz#41dc1a015e3d581f1621776be31afb2876a9b1bc" - integrity sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ== - -qs@~6.5.2: - version "6.5.2" - resolved "https://registry.yarnpkg.com/qs/-/qs-6.5.2.tgz#cb3ae806e8740444584ef154ce8ee98d403f3e36" - integrity sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA== - -querystring-es3@^0.2.0: - version "0.2.1" - resolved "https://registry.yarnpkg.com/querystring-es3/-/querystring-es3-0.2.1.tgz#9ec61f79049875707d69414596fd907a4d711e73" - integrity sha1-nsYfeQSYdXB9aUFFlv2Qek1xHnM= - -querystring@0.2.0: - version "0.2.0" - resolved "https://registry.yarnpkg.com/querystring/-/querystring-0.2.0.tgz#b209849203bb25df820da756e747005878521620" - integrity sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA= - -querystringify@^2.1.1: - version "2.2.0" - resolved "https://registry.yarnpkg.com/querystringify/-/querystringify-2.2.0.tgz#3345941b4153cb9d082d8eee4cda2016a9aef7f6" - integrity sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ== - -queue-microtask@^1.2.2: - version "1.2.2" - resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.2.tgz#abf64491e6ecf0f38a6502403d4cda04f372dfd3" - integrity sha512-dB15eXv3p2jDlbOiNLyMabYg1/sXvppd8DP2J3EOCQ0AkuSXCW2tP7mnVouVLJKgUMY6yP0kcQDVpLCN13h4Xg== - -randombytes@^2.0.0, randombytes@^2.0.1, randombytes@^2.0.5, randombytes@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.1.0.tgz#df6f84372f0270dc65cdf6291349ab7a473d4f2a" - integrity sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ== - dependencies: - safe-buffer "^5.1.0" - -randomfill@^1.0.3: - version "1.0.4" - resolved "https://registry.yarnpkg.com/randomfill/-/randomfill-1.0.4.tgz#c92196fc86ab42be983f1bf31778224931d61458" - integrity sha512-87lcbR8+MhcWcUiQ+9e+Rwx8MyR2P7qnt15ynUlbm3TU/fjbgz4GsvfSUDTemtCCtVCqb4ZcEFlyPNTh9bBTLw== - dependencies: - randombytes "^2.0.5" - safe-buffer "^5.1.0" - -range-parser@^1.2.1, range-parser@~1.2.1: - version "1.2.1" - resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.1.tgz#3cf37023d199e1c24d1a55b84800c2f3e6468031" - integrity sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg== - -raw-body@2.4.0: - version "2.4.0" - resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.4.0.tgz#a1ce6fb9c9bc356ca52e89256ab59059e13d0332" - integrity sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q== - dependencies: - bytes "3.1.0" - http-errors "1.7.2" - iconv-lite "0.4.24" - unpipe "1.0.0" - -raw-loader@4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/raw-loader/-/raw-loader-4.0.2.tgz#1aac6b7d1ad1501e66efdac1522c73e59a584eb6" - integrity sha512-ZnScIV3ag9A4wPX/ZayxL/jZH+euYb6FcUinPcgiQW0+UBtEv0O6Q3lGd3cqJ+GHH+rksEv3Pj99oxJ3u3VIKA== - dependencies: - loader-utils "^2.0.0" - schema-utils "^3.0.0" - -read-cache@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/read-cache/-/read-cache-1.0.0.tgz#e664ef31161166c9751cdbe8dbcf86b5fb58f774" - integrity sha1-5mTvMRYRZsl1HNvo28+GtftY93Q= - dependencies: - pify "^2.3.0" - -"readable-stream@1 || 2", readable-stream@^2.0.0, readable-stream@^2.0.1, readable-stream@^2.0.2, readable-stream@^2.1.5, readable-stream@^2.2.2, readable-stream@^2.3.3, readable-stream@^2.3.6, readable-stream@~2.3.6: - version "2.3.7" - resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.7.tgz#1eca1cf711aef814c04f62252a36a62f6cb23b57" - integrity sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw== - dependencies: - core-util-is "~1.0.0" - inherits "~2.0.3" - isarray "~1.0.0" - process-nextick-args "~2.0.0" - safe-buffer "~5.1.1" - string_decoder "~1.1.1" - util-deprecate "~1.0.1" - -readable-stream@^3.0.6, readable-stream@^3.6.0: - version "3.6.0" - resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.0.tgz#337bbda3adc0706bd3e024426a286d4b4b2c9198" - integrity sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA== - dependencies: - inherits "^2.0.3" - string_decoder "^1.1.1" - util-deprecate "^1.0.1" - -readdirp@^2.2.1: - version "2.2.1" - resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-2.2.1.tgz#0e87622a3325aa33e892285caf8b4e846529a525" - integrity sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ== - dependencies: - graceful-fs "^4.1.11" - micromatch "^3.1.10" - readable-stream "^2.0.2" - -readdirp@~3.5.0: - version "3.5.0" - resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-3.5.0.tgz#9ba74c019b15d365278d2e91bb8c48d7b4d42c9e" - integrity sha512-cMhu7c/8rdhkHXWsY+osBhfSy0JikwpHK/5+imo+LpeasTF8ouErHrlYkwT0++njiyuDvc7OFY5T3ukvZ8qmFQ== - dependencies: - picomatch "^2.2.1" - -reflect-metadata@^0.1.2: - version "0.1.13" - resolved "https://registry.yarnpkg.com/reflect-metadata/-/reflect-metadata-0.1.13.tgz#67ae3ca57c972a2aa1642b10fe363fe32d49dc08" - integrity sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg== - -regenerate-unicode-properties@^8.2.0: - version "8.2.0" - resolved "https://registry.yarnpkg.com/regenerate-unicode-properties/-/regenerate-unicode-properties-8.2.0.tgz#e5de7111d655e7ba60c057dbe9ff37c87e65cdec" - integrity sha512-F9DjY1vKLo/tPePDycuH3dn9H1OTPIkVD9Kz4LODu+F2C75mgjAJ7x/gwy6ZcSNRAAkhNlJSOHRe8k3p+K9WhA== - dependencies: - regenerate "^1.4.0" - -regenerate@^1.4.0: - version "1.4.2" - resolved "https://registry.yarnpkg.com/regenerate/-/regenerate-1.4.2.tgz#b9346d8827e8f5a32f7ba29637d398b69014848a" - integrity sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A== - -regenerator-runtime@0.13.7, regenerator-runtime@^0.13.4: - version "0.13.7" - resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.13.7.tgz#cac2dacc8a1ea675feaabaeb8ae833898ae46f55" - integrity sha512-a54FxoJDIr27pgf7IgeQGxmqUNYrcV338lf/6gH456HZ/PhX+5BcwHXG9ajESmwe6WRO0tAzRUrRmNONWgkrew== - -regenerator-transform@^0.14.2: - version "0.14.5" - resolved "https://registry.yarnpkg.com/regenerator-transform/-/regenerator-transform-0.14.5.tgz#c98da154683671c9c4dcb16ece736517e1b7feb4" - integrity sha512-eOf6vka5IO151Jfsw2NO9WpGX58W6wWmefK3I1zEGr0lOD0u8rwPaNqQL1aRxUaxLeKO3ArNh3VYg1KbaD+FFw== - dependencies: - "@babel/runtime" "^7.8.4" - -regex-not@^1.0.0, regex-not@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/regex-not/-/regex-not-1.0.2.tgz#1f4ece27e00b0b65e0247a6810e6a85d83a5752c" - integrity sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A== - dependencies: - extend-shallow "^3.0.2" - safe-regex "^1.1.0" - -regex-parser@^2.2.11: - version "2.2.11" - resolved "https://registry.yarnpkg.com/regex-parser/-/regex-parser-2.2.11.tgz#3b37ec9049e19479806e878cabe7c1ca83ccfe58" - integrity sha512-jbD/FT0+9MBU2XAZluI7w2OBs1RBi6p9M83nkoZayQXXU9e8Robt69FcZc7wU4eJD/YFTjn1JdCk3rbMJajz8Q== - -regexp.prototype.flags@^1.2.0: - version "1.3.1" - resolved "https://registry.yarnpkg.com/regexp.prototype.flags/-/regexp.prototype.flags-1.3.1.tgz#7ef352ae8d159e758c0eadca6f8fcb4eef07be26" - integrity sha512-JiBdRBq91WlY7uRJ0ds7R+dU02i6LKi8r3BuQhNXn+kmeLN+EfHhfjqMRis1zJxnlu88hq/4dx0P2OP3APRTOA== - dependencies: - call-bind "^1.0.2" - define-properties "^1.1.3" - -regexpu-core@^4.7.1: - version "4.7.1" - resolved "https://registry.yarnpkg.com/regexpu-core/-/regexpu-core-4.7.1.tgz#2dea5a9a07233298fbf0db91fa9abc4c6e0f8ad6" - integrity sha512-ywH2VUraA44DZQuRKzARmw6S66mr48pQVva4LBeRhcOltJ6hExvWly5ZjFLYo67xbIxb6W1q4bAGtgfEl20zfQ== - dependencies: - regenerate "^1.4.0" - regenerate-unicode-properties "^8.2.0" - regjsgen "^0.5.1" - regjsparser "^0.6.4" - unicode-match-property-ecmascript "^1.0.4" - unicode-match-property-value-ecmascript "^1.2.0" - -regjsgen@^0.5.1: - version "0.5.2" - resolved "https://registry.yarnpkg.com/regjsgen/-/regjsgen-0.5.2.tgz#92ff295fb1deecbf6ecdab2543d207e91aa33733" - integrity sha512-OFFT3MfrH90xIW8OOSyUrk6QHD5E9JOTeGodiJeBS3J6IwlgzJMNE/1bZklWz5oTg+9dCMyEetclvCVXOPoN3A== - -regjsparser@^0.6.4: - version "0.6.7" - resolved "https://registry.yarnpkg.com/regjsparser/-/regjsparser-0.6.7.tgz#c00164e1e6713c2e3ee641f1701c4b7aa0a7f86c" - integrity sha512-ib77G0uxsA2ovgiYbCVGx4Pv3PSttAx2vIwidqQzbL2U5S4Q+j00HdSAneSBuyVcMvEnTXMjiGgB+DlXozVhpQ== - dependencies: - jsesc "~0.5.0" - -remove-trailing-separator@^1.0.1: - version "1.1.0" - resolved "https://registry.yarnpkg.com/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz#c24bce2a283adad5bc3f58e0d48249b92379d8ef" - integrity sha1-wkvOKig62tW8P1jg1IJJuSN52O8= - -repeat-element@^1.1.2: - version "1.1.3" - resolved "https://registry.yarnpkg.com/repeat-element/-/repeat-element-1.1.3.tgz#782e0d825c0c5a3bb39731f84efee6b742e6b1ce" - integrity sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g== - -repeat-string@^1.6.1: - version "1.6.1" - resolved "https://registry.yarnpkg.com/repeat-string/-/repeat-string-1.6.1.tgz#8dcae470e1c88abc2d600fff4a776286da75e637" - integrity sha1-jcrkcOHIirwtYA//Sndihtp15jc= - -request@^2.87.0, request@^2.88.2: - version "2.88.2" - resolved "https://registry.yarnpkg.com/request/-/request-2.88.2.tgz#d73c918731cb5a87da047e207234146f664d12b3" - integrity sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw== - dependencies: - aws-sign2 "~0.7.0" - aws4 "^1.8.0" - caseless "~0.12.0" - combined-stream "~1.0.6" - extend "~3.0.2" - forever-agent "~0.6.1" - form-data "~2.3.2" - har-validator "~5.1.3" - http-signature "~1.2.0" - is-typedarray "~1.0.0" - isstream "~0.1.2" - json-stringify-safe "~5.0.1" - mime-types "~2.1.19" - oauth-sign "~0.9.0" - performance-now "^2.1.0" - qs "~6.5.2" - safe-buffer "^5.1.2" - tough-cookie "~2.5.0" - tunnel-agent "^0.6.0" - uuid "^3.3.2" - -require-directory@^2.1.1: - version "2.1.1" - resolved "https://registry.yarnpkg.com/require-directory/-/require-directory-2.1.1.tgz#8c64ad5fd30dab1c976e2344ffe7f792a6a6df42" - integrity sha1-jGStX9MNqxyXbiNE/+f3kqam30I= - -require-main-filename@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/require-main-filename/-/require-main-filename-2.0.0.tgz#d0b329ecc7cc0f61649f62215be69af54aa8989b" - integrity sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg== - -requires-port@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/requires-port/-/requires-port-1.0.0.tgz#925d2601d39ac485e091cf0da5c6e694dc3dcaff" - integrity sha1-kl0mAdOaxIXgkc8NpcbmlNw9yv8= - -resolve-cwd@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/resolve-cwd/-/resolve-cwd-2.0.0.tgz#00a9f7387556e27038eae232caa372a6a59b665a" - integrity sha1-AKn3OHVW4nA46uIyyqNypqWbZlo= - dependencies: - resolve-from "^3.0.0" - -resolve-from@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-3.0.0.tgz#b22c7af7d9d6881bc8b6e653335eebcb0a188748" - integrity sha1-six699nWiBvItuZTM17rywoYh0g= - -resolve-from@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-4.0.0.tgz#4abcd852ad32dd7baabfe9b40e00a36db5f392e6" - integrity sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g== - -resolve-url-loader@3.1.2: - version "3.1.2" - resolved "https://registry.yarnpkg.com/resolve-url-loader/-/resolve-url-loader-3.1.2.tgz#235e2c28e22e3e432ba7a5d4e305c59a58edfc08" - integrity sha512-QEb4A76c8Mi7I3xNKXlRKQSlLBwjUV/ULFMP+G7n3/7tJZ8MG5wsZ3ucxP1Jz8Vevn6fnJsxDx9cIls+utGzPQ== - dependencies: - adjust-sourcemap-loader "3.0.0" - camelcase "5.3.1" - compose-function "3.0.3" - convert-source-map "1.7.0" - es6-iterator "2.0.3" - loader-utils "1.2.3" - postcss "7.0.21" - rework "1.0.1" - rework-visit "1.0.0" - source-map "0.6.1" - -resolve-url@^0.2.1: - version "0.2.1" - resolved "https://registry.yarnpkg.com/resolve-url/-/resolve-url-0.2.1.tgz#2c637fe77c893afd2a663fe21aa9080068e2052a" - integrity sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo= - -resolve@1.18.1: - version "1.18.1" - resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.18.1.tgz#018fcb2c5b207d2a6424aee361c5a266da8f4130" - integrity sha512-lDfCPaMKfOJXjy0dPayzPdF1phampNWr3qFCjAu+rw/qbQmr5jWH5xN2hwh9QKfw9E5v4hwV7A+jrCmL8yjjqA== - dependencies: - is-core-module "^2.0.0" - path-parse "^1.0.6" - -resolve@^1.1.7, resolve@^1.10.0, resolve@^1.3.2, resolve@^1.8.1: - version "1.20.0" - resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.20.0.tgz#629a013fb3f70755d6f0b7935cc1c2c5378b1975" - integrity sha512-wENBPt4ySzg4ybFQW2TT1zMQucPK95HSh/nq2CFTZVOGut2+pQvSsgtda4d26YrYcr067wjbmzOG8byDPBX63A== - dependencies: - is-core-module "^2.2.0" - path-parse "^1.0.6" - -responselike@^1.0.2: - version "1.0.2" - resolved "https://registry.yarnpkg.com/responselike/-/responselike-1.0.2.tgz#918720ef3b631c5642be068f15ade5a46f4ba1e7" - integrity sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec= - dependencies: - lowercase-keys "^1.0.0" - -restore-cursor@^3.1.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-3.1.0.tgz#39f67c54b3a7a58cea5236d95cf0034239631f7e" - integrity sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA== - dependencies: - onetime "^5.1.0" - signal-exit "^3.0.2" - -ret@~0.1.10: - version "0.1.15" - resolved "https://registry.yarnpkg.com/ret/-/ret-0.1.15.tgz#b8a4825d5bdb1fc3f6f53c2bc33f81388681c7bc" - integrity sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg== - -retry@^0.10.0: - version "0.10.1" - resolved "https://registry.yarnpkg.com/retry/-/retry-0.10.1.tgz#e76388d217992c252750241d3d3956fed98d8ff4" - integrity sha1-52OI0heZLCUnUCQdPTlW/tmNj/Q= - -retry@^0.12.0: - version "0.12.0" - resolved "https://registry.yarnpkg.com/retry/-/retry-0.12.0.tgz#1b42a6266a21f07421d1b0b54b7dc167b01c013b" - integrity sha1-G0KmJmoh8HQh0bC1S33BZ7AcATs= - -reusify@^1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/reusify/-/reusify-1.0.4.tgz#90da382b1e126efc02146e90845a88db12925d76" - integrity sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw== - -rework-visit@1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/rework-visit/-/rework-visit-1.0.0.tgz#9945b2803f219e2f7aca00adb8bc9f640f842c9a" - integrity sha1-mUWygD8hni96ygCtuLyfZA+ELJo= - -rework@1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/rework/-/rework-1.0.1.tgz#30806a841342b54510aa4110850cd48534144aa7" - integrity sha1-MIBqhBNCtUUQqkEQhQzUhTQUSqc= - dependencies: - convert-source-map "^0.3.3" - css "^2.0.0" - -rfdc@^1.1.4: - version "1.2.0" - resolved "https://registry.yarnpkg.com/rfdc/-/rfdc-1.2.0.tgz#9e9894258f48f284b43c3143c68070a4f373b949" - integrity sha512-ijLyszTMmUrXvjSooucVQwimGUk84eRcmCuLV8Xghe3UO85mjUtRAHRyoMM6XtyqbECaXuBWx18La3523sXINA== - -rgb-regex@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/rgb-regex/-/rgb-regex-1.0.1.tgz#c0e0d6882df0e23be254a475e8edd41915feaeb1" - integrity sha1-wODWiC3w4jviVKR16O3UGRX+rrE= - -rgba-regex@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/rgba-regex/-/rgba-regex-1.0.0.tgz#43374e2e2ca0968b0ef1523460b7d730ff22eeb3" - integrity sha1-QzdOLiyglosO8VI0YLfXMP8i7rM= - -rimraf@3.0.2, rimraf@^3.0.0, rimraf@^3.0.2: - version "3.0.2" - resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-3.0.2.tgz#f1a5402ba6220ad52cc1282bac1ae3aa49fd061a" - integrity sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA== - dependencies: - glob "^7.1.3" - -rimraf@^2.2.8, rimraf@^2.5.2, rimraf@^2.5.4, rimraf@^2.6.2, rimraf@^2.6.3: - version "2.7.1" - resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-2.7.1.tgz#35797f13a7fdadc566142c29d4f07ccad483e3ec" - integrity sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w== - dependencies: - glob "^7.1.3" - -ripemd160@^2.0.0, ripemd160@^2.0.1: - version "2.0.2" - resolved "https://registry.yarnpkg.com/ripemd160/-/ripemd160-2.0.2.tgz#a1c1a6f624751577ba5d07914cbc92850585890c" - integrity sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA== - dependencies: - hash-base "^3.0.0" - inherits "^2.0.1" - -roarr@^2.15.3: - version "2.15.4" - resolved "https://registry.yarnpkg.com/roarr/-/roarr-2.15.4.tgz#f5fe795b7b838ccfe35dc608e0282b9eba2e7afd" - integrity sha512-CHhPh+UNHD2GTXNYhPWLnU8ONHdI+5DI+4EYIAOaiD63rHeYlZvyh8P+in5999TTSFgUYuKUAjzRI4mdh/p+2A== - dependencies: - boolean "^3.0.1" - detect-node "^2.0.4" - globalthis "^1.0.1" - json-stringify-safe "^5.0.1" - semver-compare "^1.0.0" - sprintf-js "^1.1.2" - -rollup@2.32.1: - version "2.32.1" - resolved "https://registry.yarnpkg.com/rollup/-/rollup-2.32.1.tgz#625a92c54f5b4d28ada12d618641491d4dbb548c" - integrity sha512-Op2vWTpvK7t6/Qnm1TTh7VjEZZkN8RWgf0DHbkKzQBwNf748YhXbozHVefqpPp/Fuyk/PQPAnYsBxAEtlMvpUw== - optionalDependencies: - fsevents "~2.1.2" - -run-async@^2.4.0: - version "2.4.1" - resolved "https://registry.yarnpkg.com/run-async/-/run-async-2.4.1.tgz#8440eccf99ea3e70bd409d49aab88e10c189a455" - integrity sha512-tvVnVv01b8c1RrA6Ep7JkStj85Guv/YrMcwqYQnwjsAS2cTmmPGBBjAjpCW7RrSodNSoE2/qg9O4bceNvUuDgQ== - -run-parallel@^1.1.9: - version "1.2.0" - resolved "https://registry.yarnpkg.com/run-parallel/-/run-parallel-1.2.0.tgz#66d1368da7bdf921eb9d95bd1a9229e7f21a43ee" - integrity sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA== - dependencies: - queue-microtask "^1.2.2" - -run-queue@^1.0.0, run-queue@^1.0.3: - version "1.0.3" - resolved "https://registry.yarnpkg.com/run-queue/-/run-queue-1.0.3.tgz#e848396f057d223f24386924618e25694161ec47" - integrity sha1-6Eg5bwV9Ij8kOGkkYY4laUFh7Ec= - dependencies: - aproba "^1.1.1" - -rxjs@6.6.3: - version "6.6.3" - resolved "https://registry.yarnpkg.com/rxjs/-/rxjs-6.6.3.tgz#8ca84635c4daa900c0d3967a6ee7ac60271ee552" - integrity sha512-trsQc+xYYXZ3urjOiJOuCOa5N3jAZ3eiSpQB5hIT8zGlL2QfnHLJ2r7GMkBGuIausdJN1OneaI6gQlsqNHHmZQ== - dependencies: - tslib "^1.9.0" - -rxjs@^6.5.3, rxjs@^6.6.0: - version "6.6.6" - resolved "https://registry.yarnpkg.com/rxjs/-/rxjs-6.6.6.tgz#14d8417aa5a07c5e633995b525e1e3c0dec03b70" - integrity sha512-/oTwee4N4iWzAMAL9xdGKjkEHmIwupR3oXbQjCKywF1BeFohswF3vZdogbmEF6pZkOsXTzWkrZszrWpQTByYVg== - dependencies: - tslib "^1.9.0" - -safe-buffer@5.1.2, safe-buffer@~5.1.0, safe-buffer@~5.1.1: - version "5.1.2" - resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz#991ec69d296e0313747d59bdfd2b745c35f8828d" - integrity sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g== - -safe-buffer@>=5.1.0, safe-buffer@^5.0.1, safe-buffer@^5.1.0, safe-buffer@^5.1.1, safe-buffer@^5.1.2, safe-buffer@^5.2.0, safe-buffer@~5.2.0: - version "5.2.1" - resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6" - integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== - -safe-regex@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/safe-regex/-/safe-regex-1.1.0.tgz#40a3669f3b077d1e943d44629e157dd48023bf2e" - integrity sha1-QKNmnzsHfR6UPURinhV91IAjvy4= - dependencies: - ret "~0.1.10" - -"safer-buffer@>= 2.1.2 < 3", "safer-buffer@>= 2.1.2 < 3.0.0", safer-buffer@^2.0.2, safer-buffer@^2.1.0, safer-buffer@^2.1.2, safer-buffer@~2.1.0: - version "2.1.2" - resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" - integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg== - -sass-loader@10.0.5: - version "10.0.5" - resolved "https://registry.yarnpkg.com/sass-loader/-/sass-loader-10.0.5.tgz#f53505b5ddbedf43797470ceb34066ded82bb769" - integrity sha512-2LqoNPtKkZq/XbXNQ4C64GFEleSEHKv6NPSI+bMC/l+jpEXGJhiRYkAQToO24MR7NU4JRY2RpLpJ/gjo2Uf13w== - dependencies: - klona "^2.0.4" - loader-utils "^2.0.0" - neo-async "^2.6.2" - schema-utils "^3.0.0" - semver "^7.3.2" - -sass@1.27.0: - version "1.27.0" - resolved "https://registry.yarnpkg.com/sass/-/sass-1.27.0.tgz#0657ff674206b95ec20dc638a93e179c78f6ada2" - integrity sha512-0gcrER56OkzotK/GGwgg4fPrKuiFlPNitO7eUJ18Bs+/NBlofJfMxmxqpqJxjae9vu0Wq8TZzrSyxZal00WDig== - dependencies: - chokidar ">=2.0.0 <4.0.0" - -saucelabs@^1.5.0: - version "1.5.0" - resolved "https://registry.yarnpkg.com/saucelabs/-/saucelabs-1.5.0.tgz#9405a73c360d449b232839919a86c396d379fd9d" - integrity sha512-jlX3FGdWvYf4Q3LFfFWS1QvPg3IGCGWxIc8QBFdPTbpTJnt/v17FHXYVAn7C8sHf1yUXo2c7yIM0isDryfYtHQ== - dependencies: - https-proxy-agent "^2.2.1" - -sax@>=0.6.0, sax@~1.2.4: - version "1.2.4" - resolved "https://registry.yarnpkg.com/sax/-/sax-1.2.4.tgz#2816234e2378bddc4e5354fab5caa895df7100d9" - integrity sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw== - -schema-utils@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/schema-utils/-/schema-utils-1.0.0.tgz#0b79a93204d7b600d4b2850d1f66c2a34951c770" - integrity sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g== - dependencies: - ajv "^6.1.0" - ajv-errors "^1.0.0" - ajv-keywords "^3.1.0" - -schema-utils@^2.6.5, schema-utils@^2.7.0, schema-utils@^2.7.1: - version "2.7.1" - resolved "https://registry.yarnpkg.com/schema-utils/-/schema-utils-2.7.1.tgz#1ca4f32d1b24c590c203b8e7a50bf0ea4cd394d7" - integrity sha512-SHiNtMOUGWBQJwzISiVYKu82GiV4QYGePp3odlY1tuKO7gPtphAT5R/py0fA6xtbgLL/RvtJZnU9b8s0F1q0Xg== - dependencies: - "@types/json-schema" "^7.0.5" - ajv "^6.12.4" - ajv-keywords "^3.5.2" - -schema-utils@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/schema-utils/-/schema-utils-3.0.0.tgz#67502f6aa2b66a2d4032b4279a2944978a0913ef" - integrity sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA== - dependencies: - "@types/json-schema" "^7.0.6" - ajv "^6.12.5" - ajv-keywords "^3.5.2" - -select-hose@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/select-hose/-/select-hose-2.0.0.tgz#625d8658f865af43ec962bfc376a37359a4994ca" - integrity sha1-Yl2GWPhlr0Psliv8N2o3NZpJlMo= - -selenium-webdriver@3.6.0, selenium-webdriver@^3.0.1: - version "3.6.0" - resolved "https://registry.yarnpkg.com/selenium-webdriver/-/selenium-webdriver-3.6.0.tgz#2ba87a1662c020b8988c981ae62cb2a01298eafc" - integrity sha512-WH7Aldse+2P5bbFBO4Gle/nuQOdVwpHMTL6raL3uuBj/vPG07k6uzt3aiahu352ONBr5xXh0hDlM3LhtXPOC4Q== - dependencies: - jszip "^3.1.3" - rimraf "^2.5.4" - tmp "0.0.30" - xml2js "^0.4.17" - -selfsigned@^1.10.7: - version "1.10.8" - resolved "https://registry.yarnpkg.com/selfsigned/-/selfsigned-1.10.8.tgz#0d17208b7d12c33f8eac85c41835f27fc3d81a30" - integrity sha512-2P4PtieJeEwVgTU9QEcwIRDQ/mXJLX8/+I3ur+Pg16nS8oNbrGxEso9NyYWy8NAmXiNl4dlAp5MwoNeCWzON4w== - dependencies: - node-forge "^0.10.0" - -semver-compare@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/semver-compare/-/semver-compare-1.0.0.tgz#0dee216a1c941ab37e9efb1788f6afc5ff5537fc" - integrity sha1-De4hahyUGrN+nvsXiPavxf9VN/w= - -semver-dsl@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/semver-dsl/-/semver-dsl-1.0.1.tgz#d3678de5555e8a61f629eed025366ae5f27340a0" - integrity sha1-02eN5VVeimH2Ke7QJTZq5fJzQKA= - dependencies: - semver "^5.3.0" - -semver-intersect@1.4.0: - version "1.4.0" - resolved "https://registry.yarnpkg.com/semver-intersect/-/semver-intersect-1.4.0.tgz#bdd9c06bedcdd2fedb8cd352c3c43ee8c61321f3" - integrity sha512-d8fvGg5ycKAq0+I6nfWeCx6ffaWJCsBYU0H2Rq56+/zFePYfT8mXkB3tWBSjR5BerkHNZ5eTPIk1/LBYas35xQ== - dependencies: - semver "^5.0.0" - -"semver@2 || 3 || 4 || 5", semver@^5.0.0, semver@^5.3.0, semver@^5.4.1, semver@^5.5.0, semver@^5.5.1, semver@^5.6.0: - version "5.7.1" - resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.1.tgz#a954f931aeba508d307bbf069eff0c01c96116f7" - integrity sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ== - -semver@7.0.0: - version "7.0.0" - resolved "https://registry.yarnpkg.com/semver/-/semver-7.0.0.tgz#5f3ca35761e47e05b206c6daff2cf814f0316b8e" - integrity sha512-+GB6zVA9LWh6zovYQLALHwv5rb2PHGlJi3lfiqIHxR0uuwCgefcOJc59v9fv1w8GbStwxuuqqAjI9NMAOOgq1A== - -semver@7.3.2: - version "7.3.2" - resolved "https://registry.yarnpkg.com/semver/-/semver-7.3.2.tgz#604962b052b81ed0786aae84389ffba70ffd3938" - integrity sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ== - -semver@^6.0.0, semver@^6.2.0, semver@^6.3.0: - version "6.3.0" - resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.0.tgz#ee0a64c8af5e8ceea67687b133761e1becbd1d3d" - integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw== - -semver@^7.0.0, semver@^7.1.1, semver@^7.3.2: - version "7.3.4" - resolved "https://registry.yarnpkg.com/semver/-/semver-7.3.4.tgz#27aaa7d2e4ca76452f98d3add093a72c943edc97" - integrity sha512-tCfb2WLjqFAtXn4KEdxIhalnRtoKFN7nAwj0B3ZXCbQloV2tq5eDbcTmT68JJD3nRJq24/XgxtQKFIpQdtvmVw== - dependencies: - lru-cache "^6.0.0" - -send@0.17.1: - version "0.17.1" - resolved "https://registry.yarnpkg.com/send/-/send-0.17.1.tgz#c1d8b059f7900f7466dd4938bdc44e11ddb376c8" - integrity sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg== - dependencies: - debug "2.6.9" - depd "~1.1.2" - destroy "~1.0.4" - encodeurl "~1.0.2" - escape-html "~1.0.3" - etag "~1.8.1" - fresh "0.5.2" - http-errors "~1.7.2" - mime "1.6.0" - ms "2.1.1" - on-finished "~2.3.0" - range-parser "~1.2.1" - statuses "~1.5.0" - -serialize-error@^7.0.1: - version "7.0.1" - resolved "https://registry.yarnpkg.com/serialize-error/-/serialize-error-7.0.1.tgz#f1360b0447f61ffb483ec4157c737fab7d778e18" - integrity sha512-8I8TjW5KMOKsZQTvoxjuSIa7foAwPWGOts+6o7sgjz41/qMD9VQHEDxi6PBvK2l0MXUmqZyNpUK+T2tQaaElvw== - dependencies: - type-fest "^0.13.1" - -serialize-javascript@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/serialize-javascript/-/serialize-javascript-4.0.0.tgz#b525e1238489a5ecfc42afacc3fe99e666f4b1aa" - integrity sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw== - dependencies: - randombytes "^2.1.0" - -serialize-javascript@^5.0.1: - version "5.0.1" - resolved "https://registry.yarnpkg.com/serialize-javascript/-/serialize-javascript-5.0.1.tgz#7886ec848049a462467a97d3d918ebb2aaf934f4" - integrity sha512-SaaNal9imEO737H2c05Og0/8LUXG7EnsZyMa8MzkmuHoELfT6txuj0cMqRj6zfPKnmQ1yasR4PCJc8x+M4JSPA== - dependencies: - randombytes "^2.1.0" - -serve-index@^1.9.1: - version "1.9.1" - resolved "https://registry.yarnpkg.com/serve-index/-/serve-index-1.9.1.tgz#d3768d69b1e7d82e5ce050fff5b453bea12a9239" - integrity sha1-03aNabHn2C5c4FD/9bRTvqEqkjk= - dependencies: - accepts "~1.3.4" - batch "0.6.1" - debug "2.6.9" - escape-html "~1.0.3" - http-errors "~1.6.2" - mime-types "~2.1.17" - parseurl "~1.3.2" - -serve-static@1.14.1: - version "1.14.1" - resolved "https://registry.yarnpkg.com/serve-static/-/serve-static-1.14.1.tgz#666e636dc4f010f7ef29970a88a674320898b2f9" - integrity sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg== - dependencies: - encodeurl "~1.0.2" - escape-html "~1.0.3" - parseurl "~1.3.3" - send "0.17.1" - -set-blocking@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/set-blocking/-/set-blocking-2.0.0.tgz#045f9782d011ae9a6803ddd382b24392b3d890f7" - integrity sha1-BF+XgtARrppoA93TgrJDkrPYkPc= - -set-immediate-shim@~1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/set-immediate-shim/-/set-immediate-shim-1.0.1.tgz#4b2b1b27eb808a9f8dcc481a58e5e56f599f3f61" - integrity sha1-SysbJ+uAip+NzEgaWOXlb1mfP2E= - -set-value@^2.0.0, set-value@^2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/set-value/-/set-value-2.0.1.tgz#a18d40530e6f07de4228c7defe4227af8cad005b" - integrity sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw== - dependencies: - extend-shallow "^2.0.1" - is-extendable "^0.1.1" - is-plain-object "^2.0.3" - split-string "^3.0.1" - -setimmediate@^1.0.4: - version "1.0.5" - resolved "https://registry.yarnpkg.com/setimmediate/-/setimmediate-1.0.5.tgz#290cbb232e306942d7d7ea9b83732ab7856f8285" - integrity sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU= - -setprototypeof@1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.1.0.tgz#d0bd85536887b6fe7c0d818cb962d9d91c54e656" - integrity sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ== - -setprototypeof@1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.1.1.tgz#7e95acb24aa92f5885e0abef5ba131330d4ae683" - integrity sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw== - -sha.js@^2.4.0, sha.js@^2.4.8: - version "2.4.11" - resolved "https://registry.yarnpkg.com/sha.js/-/sha.js-2.4.11.tgz#37a5cf0b81ecbc6943de109ba2960d1b26584ae7" - integrity sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ== - dependencies: - inherits "^2.0.1" - safe-buffer "^5.0.1" - -shallow-clone@^3.0.0: - version "3.0.1" - resolved "https://registry.yarnpkg.com/shallow-clone/-/shallow-clone-3.0.1.tgz#8f2981ad92531f55035b01fb230769a40e02efa3" - integrity sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA== - dependencies: - kind-of "^6.0.2" - -shebang-command@^1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-1.2.0.tgz#44aac65b695b03398968c39f363fee5deafdf1ea" - integrity sha1-RKrGW2lbAzmJaMOfNj/uXer98eo= - dependencies: - shebang-regex "^1.0.0" - -shebang-regex@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-1.0.0.tgz#da42f49740c0b42db2ca9728571cb190c98efea3" - integrity sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM= - -signal-exit@^3.0.0, signal-exit@^3.0.2: - version "3.0.3" - resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.3.tgz#a1410c2edd8f077b08b4e253c8eacfcaf057461c" - integrity sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA== - -simple-swizzle@^0.2.2: - version "0.2.2" - resolved "https://registry.yarnpkg.com/simple-swizzle/-/simple-swizzle-0.2.2.tgz#a4da6b635ffcccca33f70d17cb92592de95e557a" - integrity sha1-pNprY1/8zMoz9w0Xy5JZLeleVXo= - dependencies: - is-arrayish "^0.3.1" - -slash@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/slash/-/slash-3.0.0.tgz#6539be870c165adbd5240220dbe361f1bc4d4634" - integrity sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q== - -smart-buffer@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/smart-buffer/-/smart-buffer-4.1.0.tgz#91605c25d91652f4661ea69ccf45f1b331ca21ba" - integrity sha512-iVICrxOzCynf/SNaBQCw34eM9jROU/s5rzIhpOvzhzuYHfJR/DhZfDkXiZSgKXfgv26HT3Yni3AV/DGw0cGnnw== - -snapdragon-node@^2.0.1: - version "2.1.1" - resolved "https://registry.yarnpkg.com/snapdragon-node/-/snapdragon-node-2.1.1.tgz#6c175f86ff14bdb0724563e8f3c1b021a286853b" - integrity sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw== - dependencies: - define-property "^1.0.0" - isobject "^3.0.0" - snapdragon-util "^3.0.1" - -snapdragon-util@^3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/snapdragon-util/-/snapdragon-util-3.0.1.tgz#f956479486f2acd79700693f6f7b805e45ab56e2" - integrity sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ== - dependencies: - kind-of "^3.2.0" - -snapdragon@^0.8.1: - version "0.8.2" - resolved "https://registry.yarnpkg.com/snapdragon/-/snapdragon-0.8.2.tgz#64922e7c565b0e14204ba1aa7d6964278d25182d" - integrity sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg== - dependencies: - base "^0.11.1" - debug "^2.2.0" - define-property "^0.2.5" - extend-shallow "^2.0.1" - map-cache "^0.2.2" - source-map "^0.5.6" - source-map-resolve "^0.5.0" - use "^3.1.0" - -socket.io-adapter@~1.1.0: - version "1.1.2" - resolved "https://registry.yarnpkg.com/socket.io-adapter/-/socket.io-adapter-1.1.2.tgz#ab3f0d6f66b8fc7fca3959ab5991f82221789be9" - integrity sha512-WzZRUj1kUjrTIrUKpZLEzFZ1OLj5FwLlAFQs9kuZJzJi5DKdU7FsWc36SNmA8iDOtwBQyT8FkrriRM8vXLYz8g== - -socket.io-client@2.4.0: - version "2.4.0" - resolved "https://registry.yarnpkg.com/socket.io-client/-/socket.io-client-2.4.0.tgz#aafb5d594a3c55a34355562fc8aea22ed9119a35" - integrity sha512-M6xhnKQHuuZd4Ba9vltCLT9oa+YvTsP8j9NcEiLElfIg8KeYPyhWOes6x4t+LTAC8enQbE/995AdTem2uNyKKQ== - dependencies: - backo2 "1.0.2" - component-bind "1.0.0" - component-emitter "~1.3.0" - debug "~3.1.0" - engine.io-client "~3.5.0" - has-binary2 "~1.0.2" - indexof "0.0.1" - parseqs "0.0.6" - parseuri "0.0.6" - socket.io-parser "~3.3.0" - to-array "0.1.4" - -socket.io-parser@~3.3.0: - version "3.3.2" - resolved "https://registry.yarnpkg.com/socket.io-parser/-/socket.io-parser-3.3.2.tgz#ef872009d0adcf704f2fbe830191a14752ad50b6" - integrity sha512-FJvDBuOALxdCI9qwRrO/Rfp9yfndRtc1jSgVgV8FDraihmSP/MLGD5PEuJrNfjALvcQ+vMDM/33AWOYP/JSjDg== - dependencies: - component-emitter "~1.3.0" - debug "~3.1.0" - isarray "2.0.1" - -socket.io-parser@~3.4.0: - version "3.4.1" - resolved "https://registry.yarnpkg.com/socket.io-parser/-/socket.io-parser-3.4.1.tgz#b06af838302975837eab2dc980037da24054d64a" - integrity sha512-11hMgzL+WCLWf1uFtHSNvliI++tcRUWdoeYuwIl+Axvwy9z2gQM+7nJyN3STj1tLj5JyIUH8/gpDGxzAlDdi0A== - dependencies: - component-emitter "1.2.1" - debug "~4.1.0" - isarray "2.0.1" - -socket.io@^2.3.0: - version "2.4.1" - resolved "https://registry.yarnpkg.com/socket.io/-/socket.io-2.4.1.tgz#95ad861c9a52369d7f1a68acf0d4a1b16da451d2" - integrity sha512-Si18v0mMXGAqLqCVpTxBa8MGqriHGQh8ccEOhmsmNS3thNCGBwO8WGrwMibANsWtQQ5NStdZwHqZR3naJVFc3w== - dependencies: - debug "~4.1.0" - engine.io "~3.5.0" - has-binary2 "~1.0.2" - socket.io-adapter "~1.1.0" - socket.io-client "2.4.0" - socket.io-parser "~3.4.0" - -sockjs-client@1.4.0: - version "1.4.0" - resolved "https://registry.yarnpkg.com/sockjs-client/-/sockjs-client-1.4.0.tgz#c9f2568e19c8fd8173b4997ea3420e0bb306c7d5" - integrity sha512-5zaLyO8/nri5cua0VtOrFXBPK1jbL4+1cebT/mmKA1E1ZXOvJrII75bPu0l0k843G/+iAbhEqzyKr0w/eCCj7g== - dependencies: - debug "^3.2.5" - eventsource "^1.0.7" - faye-websocket "~0.11.1" - inherits "^2.0.3" - json3 "^3.3.2" - url-parse "^1.4.3" - -sockjs@0.3.20: - version "0.3.20" - resolved "https://registry.yarnpkg.com/sockjs/-/sockjs-0.3.20.tgz#b26a283ec562ef8b2687b44033a4eeceac75d855" - integrity sha512-SpmVOVpdq0DJc0qArhF3E5xsxvaiqGNb73XfgBpK1y3UD5gs8DSo8aCTsuT5pX8rssdc2NDIzANwP9eCAiSdTA== - dependencies: - faye-websocket "^0.10.0" - uuid "^3.4.0" - websocket-driver "0.6.5" - -socks-proxy-agent@^4.0.0: - version "4.0.2" - resolved "https://registry.yarnpkg.com/socks-proxy-agent/-/socks-proxy-agent-4.0.2.tgz#3c8991f3145b2799e70e11bd5fbc8b1963116386" - integrity sha512-NT6syHhI9LmuEMSK6Kd2V7gNv5KFZoLE7V5udWmn0de+3Mkj3UMA/AJPLyeNUVmElCurSHtUdM3ETpR3z770Wg== - dependencies: - agent-base "~4.2.1" - socks "~2.3.2" - -socks@~2.3.2: - version "2.3.3" - resolved "https://registry.yarnpkg.com/socks/-/socks-2.3.3.tgz#01129f0a5d534d2b897712ed8aceab7ee65d78e3" - integrity sha512-o5t52PCNtVdiOvzMry7wU4aOqYWL0PeCXRWBEiJow4/i/wr+wpsJQ9awEu1EonLIqsfGd5qSgDdxEOvCdmBEpA== - dependencies: - ip "1.1.5" - smart-buffer "^4.1.0" - -source-list-map@^2.0.0, source-list-map@^2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/source-list-map/-/source-list-map-2.0.1.tgz#3993bd873bfc48479cca9ea3a547835c7c154b34" - integrity sha512-qnQ7gVMxGNxsiL4lEuJwe/To8UnK7fAnmbGEEH8RpLouuKbeEm0lhbQVFIrNSuB+G7tVrAlVsZgETT5nljf+Iw== - -source-map-loader@1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/source-map-loader/-/source-map-loader-1.1.2.tgz#5b782bf08496d3a7f355e1780df0e25190a80991" - integrity sha512-bjf6eSENOYBX4JZDfl9vVLNsGAQ6Uz90fLmOazcmMcyDYOBFsGxPNn83jXezWLY9bJsVAo1ObztxPcV8HAbjVA== - dependencies: - abab "^2.0.5" - iconv-lite "^0.6.2" - loader-utils "^2.0.0" - schema-utils "^3.0.0" - source-map "^0.6.1" - whatwg-mimetype "^2.3.0" - -source-map-resolve@^0.5.0, source-map-resolve@^0.5.2: - version "0.5.3" - resolved "https://registry.yarnpkg.com/source-map-resolve/-/source-map-resolve-0.5.3.tgz#190866bece7553e1f8f267a2ee82c606b5509a1a" - integrity sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw== - dependencies: - atob "^2.1.2" - decode-uri-component "^0.2.0" - resolve-url "^0.2.1" - source-map-url "^0.4.0" - urix "^0.1.0" - -source-map-support@0.5.19, source-map-support@^0.5.17, source-map-support@^0.5.5, source-map-support@~0.5.12, source-map-support@~0.5.19: - version "0.5.19" - resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.19.tgz#a98b62f86dcaf4f67399648c085291ab9e8fed61" - integrity sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw== - dependencies: - buffer-from "^1.0.0" - source-map "^0.6.0" - -source-map-support@~0.4.0: - version "0.4.18" - resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.4.18.tgz#0286a6de8be42641338594e97ccea75f0a2c585f" - integrity sha512-try0/JqxPLF9nOjvSta7tVondkP5dwgyLDjVoyMDlmjugT2lRZ1OfsrYTkCd2hkDnJTKRbO/Rl3orm8vlsUzbA== - dependencies: - source-map "^0.5.6" - -source-map-url@^0.4.0: - version "0.4.1" - resolved "https://registry.yarnpkg.com/source-map-url/-/source-map-url-0.4.1.tgz#0af66605a745a5a2f91cf1bbf8a7afbc283dec56" - integrity sha512-cPiFOTLUKvJFIg4SKVScy4ilPPW6rFgMgfuZJPNoDuMs3nC1HbMUycBoJw77xFIp6z1UJQJOfx6C9GMH80DiTw== - -source-map@0.6.1, source-map@^0.6.0, source-map@^0.6.1, source-map@~0.6.0, source-map@~0.6.1: - version "0.6.1" - resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263" - integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g== - -source-map@0.7.3, source-map@^0.7.3, source-map@~0.7.2: - version "0.7.3" - resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.7.3.tgz#5302f8169031735226544092e64981f751750383" - integrity sha512-CkCj6giN3S+n9qrYiBTX5gystlENnRW5jZeNLHpe6aue+SrHcG5VYwujhW9s4dY31mEGsxBDrHR6oI69fTXsaQ== - -source-map@^0.5.0, source-map@^0.5.6, source-map@^0.5.7: - version "0.5.7" - resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.5.7.tgz#8a039d2d1021d22d1ea14c80d8ea468ba2ef3fcc" - integrity sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w= - -sourcemap-codec@^1.4.4, sourcemap-codec@^1.4.8: - version "1.4.8" - resolved "https://registry.yarnpkg.com/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz#ea804bd94857402e6992d05a38ef1ae35a9ab4c4" - integrity sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA== - -spdx-correct@^3.0.0: - version "3.1.1" - resolved "https://registry.yarnpkg.com/spdx-correct/-/spdx-correct-3.1.1.tgz#dece81ac9c1e6713e5f7d1b6f17d468fa53d89a9" - integrity sha512-cOYcUWwhCuHCXi49RhFRCyJEK3iPj1Ziz9DpViV3tbZOwXD49QzIN3MpOLJNxh2qwq2lJJZaKMVw9qNi4jTC0w== - dependencies: - spdx-expression-parse "^3.0.0" - spdx-license-ids "^3.0.0" - -spdx-exceptions@^2.1.0: - version "2.3.0" - resolved "https://registry.yarnpkg.com/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz#3f28ce1a77a00372683eade4a433183527a2163d" - integrity sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A== - -spdx-expression-parse@^3.0.0: - version "3.0.1" - resolved "https://registry.yarnpkg.com/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz#cf70f50482eefdc98e3ce0a6833e4a53ceeba679" - integrity sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q== - dependencies: - spdx-exceptions "^2.1.0" - spdx-license-ids "^3.0.0" - -spdx-license-ids@^3.0.0: - version "3.0.7" - resolved "https://registry.yarnpkg.com/spdx-license-ids/-/spdx-license-ids-3.0.7.tgz#e9c18a410e5ed7e12442a549fbd8afa767038d65" - integrity sha512-U+MTEOO0AiDzxwFvoa4JVnMV6mZlJKk2sBLt90s7G0Gd0Mlknc7kxEn3nuDPNZRta7O2uy8oLcZLVT+4sqNZHQ== - -spdy-transport@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/spdy-transport/-/spdy-transport-3.0.0.tgz#00d4863a6400ad75df93361a1608605e5dcdcf31" - integrity sha512-hsLVFE5SjA6TCisWeJXFKniGGOpBgMLmerfO2aCyCU5s7nJ/rpAepqmFifv/GCbSbueEeAJJnmSQ2rKC/g8Fcw== - dependencies: - debug "^4.1.0" - detect-node "^2.0.4" - hpack.js "^2.1.6" - obuf "^1.1.2" - readable-stream "^3.0.6" - wbuf "^1.7.3" - -spdy@^4.0.2: - version "4.0.2" - resolved "https://registry.yarnpkg.com/spdy/-/spdy-4.0.2.tgz#b74f466203a3eda452c02492b91fb9e84a27677b" - integrity sha512-r46gZQZQV+Kl9oItvl1JZZqJKGr+oEkB08A6BzkiR7593/7IbtuncXHd2YoYeTsG4157ZssMu9KYvUHLcjcDoA== - dependencies: - debug "^4.1.0" - handle-thing "^2.0.0" - http-deceiver "^1.2.7" - select-hose "^2.0.0" - spdy-transport "^3.0.0" - -speed-measure-webpack-plugin@1.3.3: - version "1.3.3" - resolved "https://registry.yarnpkg.com/speed-measure-webpack-plugin/-/speed-measure-webpack-plugin-1.3.3.tgz#6ff894fc83e8a6310dde3af863a0329cd79da4f5" - integrity sha512-2ljD4Ch/rz2zG3HsLsnPfp23osuPBS0qPuz9sGpkNXTN1Ic4M+W9xB8l8rS8ob2cO4b1L+WTJw/0AJwWYVgcxQ== - dependencies: - chalk "^2.0.1" - -split-string@^3.0.1, split-string@^3.0.2: - version "3.1.0" - resolved "https://registry.yarnpkg.com/split-string/-/split-string-3.1.0.tgz#7cb09dda3a86585705c64b39a6466038682e8fe2" - integrity sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw== - dependencies: - extend-shallow "^3.0.0" - -sprintf-js@^1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.1.2.tgz#da1765262bf8c0f571749f2ad6c26300207ae673" - integrity sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug== - -sprintf-js@~1.0.2: - version "1.0.3" - resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.0.3.tgz#04e6926f662895354f3dd015203633b857297e2c" - integrity sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw= - -sshpk@^1.7.0: - version "1.16.1" - resolved "https://registry.yarnpkg.com/sshpk/-/sshpk-1.16.1.tgz#fb661c0bef29b39db40769ee39fa70093d6f6877" - integrity sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg== - dependencies: - asn1 "~0.2.3" - assert-plus "^1.0.0" - bcrypt-pbkdf "^1.0.0" - dashdash "^1.12.0" - ecc-jsbn "~0.1.1" - getpass "^0.1.1" - jsbn "~0.1.0" - safer-buffer "^2.0.2" - tweetnacl "~0.14.0" - -ssri@^6.0.0, ssri@^6.0.1: - version "6.0.1" - resolved "https://registry.yarnpkg.com/ssri/-/ssri-6.0.1.tgz#2a3c41b28dd45b62b63676ecb74001265ae9edd8" - integrity sha512-3Wge10hNcT1Kur4PDFwEieXSCMCJs/7WvSACcrMYrNp+b8kDL1/0wJch5Ni2WrtwEa2IO8OsVfeKIciKCDx/QA== - dependencies: - figgy-pudding "^3.5.1" - -ssri@^8.0.0: - version "8.0.1" - resolved "https://registry.yarnpkg.com/ssri/-/ssri-8.0.1.tgz#638e4e439e2ffbd2cd289776d5ca457c4f51a2af" - integrity sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ== - dependencies: - minipass "^3.1.1" - -stable@^0.1.8: - version "0.1.8" - resolved "https://registry.yarnpkg.com/stable/-/stable-0.1.8.tgz#836eb3c8382fe2936feaf544631017ce7d47a3cf" - integrity sha512-ji9qxRnOVfcuLDySj9qzhGSEFVobyt1kIOSkj1qZzYLzq7Tos/oUUWvotUPQLlrsidqsK6tBH89Bc9kL5zHA6w== - -static-extend@^0.1.1: - version "0.1.2" - resolved "https://registry.yarnpkg.com/static-extend/-/static-extend-0.1.2.tgz#60809c39cbff55337226fd5e0b520f341f1fb5c6" - integrity sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY= - dependencies: - define-property "^0.2.5" - object-copy "^0.1.0" - -"statuses@>= 1.4.0 < 2", "statuses@>= 1.5.0 < 2", statuses@~1.5.0: - version "1.5.0" - resolved "https://registry.yarnpkg.com/statuses/-/statuses-1.5.0.tgz#161c7dac177659fd9811f43771fa99381478628c" - integrity sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow= - -stream-browserify@^2.0.1: - version "2.0.2" - resolved "https://registry.yarnpkg.com/stream-browserify/-/stream-browserify-2.0.2.tgz#87521d38a44aa7ee91ce1cd2a47df0cb49dd660b" - integrity sha512-nX6hmklHs/gr2FuxYDltq8fJA1GDlxKQCz8O/IM4atRqBH8OORmBNgfvW5gG10GT/qQ9u0CzIvr2X5Pkt6ntqg== - dependencies: - inherits "~2.0.1" - readable-stream "^2.0.2" - -stream-each@^1.1.0: - version "1.2.3" - resolved "https://registry.yarnpkg.com/stream-each/-/stream-each-1.2.3.tgz#ebe27a0c389b04fbcc233642952e10731afa9bae" - integrity sha512-vlMC2f8I2u/bZGqkdfLQW/13Zihpej/7PmSiMQsbYddxuTsJp8vRe2x2FvVExZg7FaOds43ROAuFJwPR4MTZLw== - dependencies: - end-of-stream "^1.1.0" - stream-shift "^1.0.0" - -stream-http@^2.7.2: - version "2.8.3" - resolved "https://registry.yarnpkg.com/stream-http/-/stream-http-2.8.3.tgz#b2d242469288a5a27ec4fe8933acf623de6514fc" - integrity sha512-+TSkfINHDo4J+ZobQLWiMouQYB+UVYFttRA94FpEzzJ7ZdqcL4uUUQ7WkdkI4DSozGmgBUE/a47L+38PenXhUw== - dependencies: - builtin-status-codes "^3.0.0" - inherits "^2.0.1" - readable-stream "^2.3.6" - to-arraybuffer "^1.0.0" - xtend "^4.0.0" - -stream-shift@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/stream-shift/-/stream-shift-1.0.1.tgz#d7088281559ab2778424279b0877da3c392d5a3d" - integrity sha512-AiisoFqQ0vbGcZgQPY1cdP2I76glaVA/RauYR4G4thNFgkTqr90yXTo4LYX60Jl+sIlPNHHdGSwo01AvbKUSVQ== - -streamroller@^2.2.4: - version "2.2.4" - resolved "https://registry.yarnpkg.com/streamroller/-/streamroller-2.2.4.tgz#c198ced42db94086a6193608187ce80a5f2b0e53" - integrity sha512-OG79qm3AujAM9ImoqgWEY1xG4HX+Lw+yY6qZj9R1K2mhF5bEmQ849wvrb+4vt4jLMLzwXttJlQbOdPOQVRv7DQ== - dependencies: - date-format "^2.1.0" - debug "^4.1.1" - fs-extra "^8.1.0" - -string-width@^3.0.0, string-width@^3.1.0: - version "3.1.0" - resolved "https://registry.yarnpkg.com/string-width/-/string-width-3.1.0.tgz#22767be21b62af1081574306f69ac51b62203961" - integrity sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w== - dependencies: - emoji-regex "^7.0.1" - is-fullwidth-code-point "^2.0.0" - strip-ansi "^5.1.0" - -string-width@^4.1.0, string-width@^4.2.0: - version "4.2.2" - resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.2.tgz#dafd4f9559a7585cfba529c6a0a4f73488ebd4c5" - integrity sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA== - dependencies: - emoji-regex "^8.0.0" - is-fullwidth-code-point "^3.0.0" - strip-ansi "^6.0.0" - -string.prototype.trimend@^1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/string.prototype.trimend/-/string.prototype.trimend-1.0.4.tgz#e75ae90c2942c63504686c18b287b4a0b1a45f80" - integrity sha512-y9xCjw1P23Awk8EvTpcyL2NIr1j7wJ39f+k6lvRnSMz+mz9CGz9NYPelDk42kOz6+ql8xjfK8oYzy3jAP5QU5A== - dependencies: - call-bind "^1.0.2" - define-properties "^1.1.3" - -string.prototype.trimstart@^1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/string.prototype.trimstart/-/string.prototype.trimstart-1.0.4.tgz#b36399af4ab2999b4c9c648bd7a3fb2bb26feeed" - integrity sha512-jh6e984OBfvxS50tdY2nRZnoC5/mLFKOREQfw8t5yytkoUsJRNxvI/E39qu1sD0OtWI3OC0XgKSmcWwziwYuZw== - dependencies: - call-bind "^1.0.2" - define-properties "^1.1.3" - -string_decoder@^1.0.0, string_decoder@^1.1.1: - version "1.3.0" - resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e" - integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA== - dependencies: - safe-buffer "~5.2.0" - -string_decoder@~1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.1.1.tgz#9cf1611ba62685d7030ae9e4ba34149c3af03fc8" - integrity sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg== - dependencies: - safe-buffer "~5.1.0" - -strip-ansi@^3.0.0, strip-ansi@^3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-3.0.1.tgz#6a385fb8853d952d5ff05d0e8aaf94278dc63dcf" - integrity sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8= - dependencies: - ansi-regex "^2.0.0" - -strip-ansi@^5.0.0, strip-ansi@^5.1.0, strip-ansi@^5.2.0: - version "5.2.0" - resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-5.2.0.tgz#8c9a536feb6afc962bdfa5b104a5091c1ad9c0ae" - integrity sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA== - dependencies: - ansi-regex "^4.1.0" - -strip-ansi@^6.0.0: - version "6.0.0" - resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.0.tgz#0b1571dd7669ccd4f3e06e14ef1eed26225ae532" - integrity sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w== - dependencies: - ansi-regex "^5.0.0" - -strip-eof@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/strip-eof/-/strip-eof-1.0.0.tgz#bb43ff5598a6eb05d89b59fcd129c983313606bf" - integrity sha1-u0P/VZim6wXYm1n80SnJgzE2Br8= - -style-loader@2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/style-loader/-/style-loader-2.0.0.tgz#9669602fd4690740eaaec137799a03addbbc393c" - integrity sha512-Z0gYUJmzZ6ZdRUqpg1r8GsaFKypE+3xAzuFeMuoHgjc9KZv3wMyCRjQIWEbhoFSq7+7yoHXySDJyyWQaPajeiQ== - dependencies: - loader-utils "^2.0.0" - schema-utils "^3.0.0" - -stylehacks@^4.0.0: - version "4.0.3" - resolved "https://registry.yarnpkg.com/stylehacks/-/stylehacks-4.0.3.tgz#6718fcaf4d1e07d8a1318690881e8d96726a71d5" - integrity sha512-7GlLk9JwlElY4Y6a/rmbH2MhVlTyVmiJd1PfTCqFaIBEGMYNsrO/v3SeGTdhBThLg4Z+NbOk/qFMwCa+J+3p/g== - dependencies: - browserslist "^4.0.0" - postcss "^7.0.0" - postcss-selector-parser "^3.0.0" - -stylus-loader@4.3.1: - version "4.3.1" - resolved "https://registry.yarnpkg.com/stylus-loader/-/stylus-loader-4.3.1.tgz#8b4e749294d9fe0729c2e5e1f04cbf87e1c941aa" - integrity sha512-apDYJEM5ZpOAWbWInWcsbtI8gHNr/XYVcSY/tWqOUPt7M5tqhtwXVsAkgyiVjhuvw2Yrjq474a9H+g4d047Ebw== - dependencies: - fast-glob "^3.2.4" - klona "^2.0.4" - loader-utils "^2.0.0" - normalize-path "^3.0.0" - schema-utils "^3.0.0" - -stylus@0.54.8: - version "0.54.8" - resolved "https://registry.yarnpkg.com/stylus/-/stylus-0.54.8.tgz#3da3e65966bc567a7b044bfe0eece653e099d147" - integrity sha512-vr54Or4BZ7pJafo2mpf0ZcwA74rpuYCZbxrHBsH8kbcXOwSfvBFwsRfpGO5OD5fhG5HDCFW737PKaawI7OqEAg== - dependencies: - css-parse "~2.0.0" - debug "~3.1.0" - glob "^7.1.6" - mkdirp "~1.0.4" - safer-buffer "^2.1.2" - sax "~1.2.4" - semver "^6.3.0" - source-map "^0.7.3" - -sumchecker@^3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/sumchecker/-/sumchecker-3.0.1.tgz#6377e996795abb0b6d348e9b3e1dfb24345a8e42" - integrity sha512-MvjXzkz/BOfyVDkG0oFOtBxHX2u3gKbMHIF/dXblZsgD3BWOFLmHovIpZY7BykJdAjcqRCBi1WYBNdEC9yI7vg== - dependencies: - debug "^4.1.0" - -supports-color@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-2.0.0.tgz#535d045ce6b6363fa40117084629995e9df324c7" - integrity sha1-U10EXOa2Nj+kARcIRimZXp3zJMc= - -supports-color@^5.3.0: - version "5.5.0" - resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.5.0.tgz#e2e69a44ac8772f78a1ec0b35b689df6530efc8f" - integrity sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow== - dependencies: - has-flag "^3.0.0" - -supports-color@^6.1.0: - version "6.1.0" - resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-6.1.0.tgz#0764abc69c63d5ac842dd4867e8d025e880df8f3" - integrity sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ== - dependencies: - has-flag "^3.0.0" - -supports-color@^7.0.0, supports-color@^7.1.0: - version "7.2.0" - resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da" - integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw== - dependencies: - has-flag "^4.0.0" - -svgo@^1.0.0: - version "1.3.2" - resolved "https://registry.yarnpkg.com/svgo/-/svgo-1.3.2.tgz#b6dc511c063346c9e415b81e43401145b96d4167" - integrity sha512-yhy/sQYxR5BkC98CY7o31VGsg014AKLEPxdfhora76l36hD9Rdy5NZA/Ocn6yayNPgSamYdtX2rFJdcv07AYVw== - dependencies: - chalk "^2.4.1" - coa "^2.0.2" - css-select "^2.0.0" - css-select-base-adapter "^0.1.1" - css-tree "1.0.0-alpha.37" - csso "^4.0.2" - js-yaml "^3.13.1" - mkdirp "~0.5.1" - object.values "^1.1.0" - sax "~1.2.4" - stable "^0.1.8" - unquote "~1.1.1" - util.promisify "~1.0.0" - -symbol-observable@2.0.3: - version "2.0.3" - resolved "https://registry.yarnpkg.com/symbol-observable/-/symbol-observable-2.0.3.tgz#5b521d3d07a43c351055fa43b8355b62d33fd16a" - integrity sha512-sQV7phh2WCYAn81oAkakC5qjq2Ml0g8ozqz03wOGnx9dDlG1de6yrF+0RAzSJD8fPUow3PTSMf2SAbOGxb93BA== - -tapable@^1.0.0, tapable@^1.1.3: - version "1.1.3" - resolved "https://registry.yarnpkg.com/tapable/-/tapable-1.1.3.tgz#a1fccc06b58db61fd7a45da2da44f5f3a3e67ba2" - integrity sha512-4WK/bYZmj8xLr+HUCODHGF1ZFzsYffasLUgEiMBY4fgtltdO6B4WJtlSbPaDTLpYTcGVwM2qLnFTICEcNxs3kA== - -tapable@^2.0.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/tapable/-/tapable-2.2.0.tgz#5c373d281d9c672848213d0e037d1c4165ab426b" - integrity sha512-FBk4IesMV1rBxX2tfiK8RAmogtWn53puLOQlvO8XuwlgxcYbP4mVPS9Ph4aeamSyyVjOl24aYWAuc8U5kCVwMw== - -tar@^4.4.10: - version "4.4.13" - resolved "https://registry.yarnpkg.com/tar/-/tar-4.4.13.tgz#43b364bc52888d555298637b10d60790254ab525" - integrity sha512-w2VwSrBoHa5BsSyH+KxEqeQBAllHhccyMFVHtGtdMpF4W7IRWfZjFiQceJPChOeTsSDVUpER2T8FA93pr0L+QA== - dependencies: - chownr "^1.1.1" - fs-minipass "^1.2.5" - minipass "^2.8.6" - minizlib "^1.2.1" - mkdirp "^0.5.0" - safe-buffer "^5.1.2" - yallist "^3.0.3" - -tar@^6.0.2: - version "6.1.0" - resolved "https://registry.yarnpkg.com/tar/-/tar-6.1.0.tgz#d1724e9bcc04b977b18d5c573b333a2207229a83" - integrity sha512-DUCttfhsnLCjwoDoFcI+B2iJgYa93vBnDUATYEeRx6sntCTdN01VnqsIuTlALXla/LWooNg0yEGeB+Y8WdFxGA== - dependencies: - chownr "^2.0.0" - fs-minipass "^2.0.0" - minipass "^3.0.0" - minizlib "^2.1.1" - mkdirp "^1.0.3" - yallist "^4.0.0" - -terser-webpack-plugin@4.2.3: - version "4.2.3" - resolved "https://registry.yarnpkg.com/terser-webpack-plugin/-/terser-webpack-plugin-4.2.3.tgz#28daef4a83bd17c1db0297070adc07fc8cfc6a9a" - integrity sha512-jTgXh40RnvOrLQNgIkwEKnQ8rmHjHK4u+6UBEi+W+FPmvb+uo+chJXntKe7/3lW5mNysgSWD60KyesnhW8D6MQ== - dependencies: - cacache "^15.0.5" - find-cache-dir "^3.3.1" - jest-worker "^26.5.0" - p-limit "^3.0.2" - schema-utils "^3.0.0" - serialize-javascript "^5.0.1" - source-map "^0.6.1" - terser "^5.3.4" - webpack-sources "^1.4.3" - -terser-webpack-plugin@^1.4.3: - version "1.4.5" - resolved "https://registry.yarnpkg.com/terser-webpack-plugin/-/terser-webpack-plugin-1.4.5.tgz#a217aefaea330e734ffacb6120ec1fa312d6040b" - integrity sha512-04Rfe496lN8EYruwi6oPQkG0vo8C+HT49X687FZnpPF0qMAIHONI6HEXYPKDOE8e5HjXTyKfqRd/agHtH0kOtw== - dependencies: - cacache "^12.0.2" - find-cache-dir "^2.1.0" - is-wsl "^1.1.0" - schema-utils "^1.0.0" - serialize-javascript "^4.0.0" - source-map "^0.6.1" - terser "^4.1.2" - webpack-sources "^1.4.0" - worker-farm "^1.7.0" - -terser@5.3.7: - version "5.3.7" - resolved "https://registry.yarnpkg.com/terser/-/terser-5.3.7.tgz#798a4ae2e7ff67050c3e99fcc4e00725827d97e2" - integrity sha512-lJbKdfxWvjpV330U4PBZStCT9h3N9A4zZVA5Y4k9sCWXknrpdyxi1oMsRKLmQ/YDMDxSBKIh88v0SkdhdqX06w== - dependencies: - commander "^2.20.0" - source-map "~0.7.2" - source-map-support "~0.5.19" - -terser@^4.1.2: - version "4.8.0" - resolved "https://registry.yarnpkg.com/terser/-/terser-4.8.0.tgz#63056343d7c70bb29f3af665865a46fe03a0df17" - integrity sha512-EAPipTNeWsb/3wLPeup1tVPaXfIaU68xMnVdPafIL1TV05OhASArYyIfFvnvJCNrR2NIOvDVNNTFRa+Re2MWyw== - dependencies: - commander "^2.20.0" - source-map "~0.6.1" - source-map-support "~0.5.12" - -terser@^5.3.4: - version "5.6.0" - resolved "https://registry.yarnpkg.com/terser/-/terser-5.6.0.tgz#138cdf21c5e3100b1b3ddfddf720962f88badcd2" - integrity sha512-vyqLMoqadC1uR0vywqOZzriDYzgEkNJFK4q9GeyOBHIbiECHiWLKcWfbQWAUaPfxkjDhapSlZB9f7fkMrvkVjA== - dependencies: - commander "^2.20.0" - source-map "~0.7.2" - source-map-support "~0.5.19" - -text-table@0.2.0: - version "0.2.0" - resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4" - integrity sha1-f17oI66AUgfACvLfSoTsP8+lcLQ= - -through2@^2.0.0: - version "2.0.5" - resolved "https://registry.yarnpkg.com/through2/-/through2-2.0.5.tgz#01c1e39eb31d07cb7d03a96a70823260b23132cd" - integrity sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ== - dependencies: - readable-stream "~2.3.6" - xtend "~4.0.1" - -"through@>=2.2.7 <3", through@X.X.X, through@^2.3.6: - version "2.3.8" - resolved "https://registry.yarnpkg.com/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5" - integrity sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU= - -thunky@^1.0.2: - version "1.1.0" - resolved "https://registry.yarnpkg.com/thunky/-/thunky-1.1.0.tgz#5abaf714a9405db0504732bbccd2cedd9ef9537d" - integrity sha512-eHY7nBftgThBqOyHGVN+l8gF0BucP09fMo0oO/Lb0w1OF80dJv+lDVpXG60WMQvkcxAkNybKsrEIE3ZtKGmPrA== - -timers-browserify@^2.0.4: - version "2.0.12" - resolved "https://registry.yarnpkg.com/timers-browserify/-/timers-browserify-2.0.12.tgz#44a45c11fbf407f34f97bccd1577c652361b00ee" - integrity sha512-9phl76Cqm6FhSX9Xe1ZUAMLtm1BLkKj2Qd5ApyWkXzsMRaA7dgr81kf4wJmQf/hAvg8EEyJxDo3du/0KlhPiKQ== - dependencies: - setimmediate "^1.0.4" - -timsort@^0.3.0: - version "0.3.0" - resolved "https://registry.yarnpkg.com/timsort/-/timsort-0.3.0.tgz#405411a8e7e6339fe64db9a234de11dc31e02bd4" - integrity sha1-QFQRqOfmM5/mTbmiNN4R3DHgK9Q= - -tmp@0.0.30: - version "0.0.30" - resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.0.30.tgz#72419d4a8be7d6ce75148fd8b324e593a711c2ed" - integrity sha1-ckGdSovn1s51FI/YsyTlk6cRwu0= - dependencies: - os-tmpdir "~1.0.1" - -tmp@0.2.1: - version "0.2.1" - resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.2.1.tgz#8457fc3037dcf4719c251367a1af6500ee1ccf14" - integrity sha512-76SUhtfqR2Ijn+xllcI5P1oyannHNHByD80W1q447gU3mp9G9PSpGdWmjUOHRDPiHYacIk66W7ubDTuPF3BEtQ== - dependencies: - rimraf "^3.0.0" - -tmp@^0.0.33: - version "0.0.33" - resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.0.33.tgz#6d34335889768d21b2bcda0aa277ced3b1bfadf9" - integrity sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw== - dependencies: - os-tmpdir "~1.0.2" - -to-array@0.1.4: - version "0.1.4" - resolved "https://registry.yarnpkg.com/to-array/-/to-array-0.1.4.tgz#17e6c11f73dd4f3d74cda7a4ff3238e9ad9bf890" - integrity sha1-F+bBH3PdTz10zaek/zI46a2b+JA= - -to-arraybuffer@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/to-arraybuffer/-/to-arraybuffer-1.0.1.tgz#7d229b1fcc637e466ca081180836a7aabff83f43" - integrity sha1-fSKbH8xjfkZsoIEYCDanqr/4P0M= - -to-fast-properties@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/to-fast-properties/-/to-fast-properties-2.0.0.tgz#dc5e698cbd079265bc73e0377681a4e4e83f616e" - integrity sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4= - -to-object-path@^0.3.0: - version "0.3.0" - resolved "https://registry.yarnpkg.com/to-object-path/-/to-object-path-0.3.0.tgz#297588b7b0e7e0ac08e04e672f85c1f4999e17af" - integrity sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68= - dependencies: - kind-of "^3.0.2" - -to-readable-stream@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/to-readable-stream/-/to-readable-stream-1.0.0.tgz#ce0aa0c2f3df6adf852efb404a783e77c0475771" - integrity sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q== - -to-regex-range@^2.1.0: - version "2.1.1" - resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-2.1.1.tgz#7c80c17b9dfebe599e27367e0d4dd5590141db38" - integrity sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg= - dependencies: - is-number "^3.0.0" - repeat-string "^1.6.1" - -to-regex-range@^5.0.1: - version "5.0.1" - resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4" - integrity sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ== - dependencies: - is-number "^7.0.0" - -to-regex@^3.0.1, to-regex@^3.0.2: - version "3.0.2" - resolved "https://registry.yarnpkg.com/to-regex/-/to-regex-3.0.2.tgz#13cfdd9b336552f30b51f33a8ae1b42a7a7599ce" - integrity sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw== - dependencies: - define-property "^2.0.2" - extend-shallow "^3.0.2" - regex-not "^1.0.2" - safe-regex "^1.1.0" - -toidentifier@1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/toidentifier/-/toidentifier-1.0.0.tgz#7e1be3470f1e77948bc43d94a3c8f4d7752ba553" - integrity sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw== - -tough-cookie@~2.5.0: - version "2.5.0" - resolved "https://registry.yarnpkg.com/tough-cookie/-/tough-cookie-2.5.0.tgz#cd9fb2a0aa1d5a12b473bd9fb96fa3dcff65ade2" - integrity sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g== - dependencies: - psl "^1.1.28" - punycode "^2.1.1" - -tree-kill@1.2.2: - version "1.2.2" - resolved "https://registry.yarnpkg.com/tree-kill/-/tree-kill-1.2.2.tgz#4ca09a9092c88b73a7cdc5e8a01b507b0790a0cc" - integrity sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A== - -ts-node@8.10.2: - version "8.10.2" - resolved "https://registry.yarnpkg.com/ts-node/-/ts-node-8.10.2.tgz#eee03764633b1234ddd37f8db9ec10b75ec7fb8d" - integrity sha512-ISJJGgkIpDdBhWVu3jufsWpK3Rzo7bdiIXJjQc0ynKxVOVcg2oIrf2H2cejminGrptVc6q6/uynAHNCuWGbpVA== - dependencies: - arg "^4.1.0" - diff "^4.0.1" - make-error "^1.1.1" - source-map-support "^0.5.17" - yn "3.1.1" - -ts-pnp@^1.1.6: - version "1.2.0" - resolved "https://registry.yarnpkg.com/ts-pnp/-/ts-pnp-1.2.0.tgz#a500ad084b0798f1c3071af391e65912c86bca92" - integrity sha512-csd+vJOb/gkzvcCHgTGSChYpy5f1/XKNsmvBGO4JXS+z1v2HobugDz4s1IeFXM3wZB44uczs+eazB5Q/ccdhQw== - -tslib@2.0.3: - version "2.0.3" - resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.0.3.tgz#8e0741ac45fc0c226e58a17bfc3e64b9bc6ca61c" - integrity sha512-uZtkfKblCEQtZKBF6EBXVZeQNl82yqtDQdv+eck8u7tdPxjLu2/lp5/uPW+um2tpuxINHWy3GhiccY7QgEaVHQ== - -tslib@^1.10.0, tslib@^1.13.0, tslib@^1.8.1, tslib@^1.9.0: - version "1.14.1" - resolved "https://registry.yarnpkg.com/tslib/-/tslib-1.14.1.tgz#cf2d38bdc34a134bcaf1091c41f6619e2f672d00" - integrity sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg== - -tslib@^2.0.0, tslib@^2.0.1: - version "2.1.0" - resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.1.0.tgz#da60860f1c2ecaa5703ab7d39bc05b6bf988b97a" - integrity sha512-hcVC3wYEziELGGmEEXue7D75zbwIIVUMWAVbHItGPx0ziyXxrOMQx4rQEVEV45Ut/1IotuEvwqPopzIOkDMf0A== - -tslint@~6.1.3: - version "6.1.3" - resolved "https://registry.yarnpkg.com/tslint/-/tslint-6.1.3.tgz#5c23b2eccc32487d5523bd3a470e9aa31789d904" - integrity sha512-IbR4nkT96EQOvKE2PW/djGz8iGNeJ4rF2mBfiYaR/nvUWYKJhLwimoJKgjIFEIDibBtOevj7BqCRL4oHeWWUCg== - dependencies: - "@babel/code-frame" "^7.0.0" - builtin-modules "^1.1.1" - chalk "^2.3.0" - commander "^2.12.1" - diff "^4.0.1" - glob "^7.1.1" - js-yaml "^3.13.1" - minimatch "^3.0.4" - mkdirp "^0.5.3" - resolve "^1.3.2" - semver "^5.3.0" - tslib "^1.13.0" - tsutils "^2.29.0" - -tsutils@^2.29.0: - version "2.29.0" - resolved "https://registry.yarnpkg.com/tsutils/-/tsutils-2.29.0.tgz#32b488501467acbedd4b85498673a0812aca0b99" - integrity sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA== - dependencies: - tslib "^1.8.1" - -tty-browserify@0.0.0: - version "0.0.0" - resolved "https://registry.yarnpkg.com/tty-browserify/-/tty-browserify-0.0.0.tgz#a157ba402da24e9bf957f9aa69d524eed42901a6" - integrity sha1-oVe6QC2iTpv5V/mqadUk7tQpAaY= - -tunnel-agent@^0.6.0: - version "0.6.0" - resolved "https://registry.yarnpkg.com/tunnel-agent/-/tunnel-agent-0.6.0.tgz#27a5dea06b36b04a0a9966774b290868f0fc40fd" - integrity sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0= - dependencies: - safe-buffer "^5.0.1" - -tunnel@^0.0.6: - version "0.0.6" - resolved "https://registry.yarnpkg.com/tunnel/-/tunnel-0.0.6.tgz#72f1314b34a5b192db012324df2cc587ca47f92c" - integrity sha512-1h/Lnq9yajKY2PEbBadPXj3VxsDDu844OnaAo52UVmIzIvwwtBPIuNvkjuzBlTWpfJyUbG3ez0KSBibQkj4ojg== - -tweetnacl@^0.14.3, tweetnacl@~0.14.0: - version "0.14.5" - resolved "https://registry.yarnpkg.com/tweetnacl/-/tweetnacl-0.14.5.tgz#5ae68177f192d4456269d108afa93ff8743f4f64" - integrity sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q= - -type-fest@^0.11.0: - version "0.11.0" - resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.11.0.tgz#97abf0872310fed88a5c466b25681576145e33f1" - integrity sha512-OdjXJxnCN1AvyLSzeKIgXTXxV+99ZuXl3Hpo9XpJAv9MBcHrrJOQ5kV7ypXOuQie+AmWG25hLbiKdwYTifzcfQ== - -type-fest@^0.13.1: - version "0.13.1" - resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.13.1.tgz#0172cb5bce80b0bd542ea348db50c7e21834d934" - integrity sha512-34R7HTnG0XIJcBSn5XhDd7nNFPRcXYRZrBB2O2jdKqYODldSzBAqzsWoZYYvduky73toYS/ESqxPvkDf/F0XMg== - -type-is@~1.6.17, type-is@~1.6.18: - version "1.6.18" - resolved "https://registry.yarnpkg.com/type-is/-/type-is-1.6.18.tgz#4e552cd05df09467dcbc4ef739de89f2cf37c131" - integrity sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g== - dependencies: - media-typer "0.3.0" - mime-types "~2.1.24" - -type@^1.0.1: - version "1.2.0" - resolved "https://registry.yarnpkg.com/type/-/type-1.2.0.tgz#848dd7698dafa3e54a6c479e759c4bc3f18847a0" - integrity sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg== - -type@^2.0.0: - version "2.5.0" - resolved "https://registry.yarnpkg.com/type/-/type-2.5.0.tgz#0a2e78c2e77907b252abe5f298c1b01c63f0db3d" - integrity sha512-180WMDQaIMm3+7hGXWf12GtdniDEy7nYcyFMKJn/eZz/6tSLXrUN9V0wKSbMjej0I1WHWbpREDEKHtqPQa9NNw== - -typedarray@^0.0.6: - version "0.0.6" - resolved "https://registry.yarnpkg.com/typedarray/-/typedarray-0.0.6.tgz#867ac74e3864187b1d3d47d996a78ec5c8830777" - integrity sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c= - -typescript@4.0.5: - version "4.0.5" - resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.0.5.tgz#ae9dddfd1069f1cb5beb3ef3b2170dd7c1332389" - integrity sha512-ywmr/VrTVCmNTJ6iV2LwIrfG1P+lv6luD8sUJs+2eI9NLGigaN+nUQc13iHqisq7bra9lnmUSYqbJvegraBOPQ== - -ua-parser-js@0.7.21: - version "0.7.21" - resolved "https://registry.yarnpkg.com/ua-parser-js/-/ua-parser-js-0.7.21.tgz#853cf9ce93f642f67174273cc34565ae6f308777" - integrity sha512-+O8/qh/Qj8CgC6eYBVBykMrNtp5Gebn4dlGD/kKXVkJNDwyrAwSIqwz8CDf+tsAIWVycKcku6gIXJ0qwx/ZXaQ== - -unbox-primitive@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/unbox-primitive/-/unbox-primitive-1.0.0.tgz#eeacbc4affa28e9b3d36b5eaeccc50b3251b1d3f" - integrity sha512-P/51NX+JXyxK/aigg1/ZgyccdAxm5K1+n8+tvqSntjOivPt19gvm1VC49RWYetsiub8WViUchdxl/KWHHB0kzA== - dependencies: - function-bind "^1.1.1" - has-bigints "^1.0.0" - has-symbols "^1.0.0" - which-boxed-primitive "^1.0.1" - -unicode-canonical-property-names-ecmascript@^1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-1.0.4.tgz#2619800c4c825800efdd8343af7dd9933cbe2818" - integrity sha512-jDrNnXWHd4oHiTZnx/ZG7gtUTVp+gCcTTKr8L0HjlwphROEW3+Him+IpvC+xcJEFegapiMZyZe02CyuOnRmbnQ== - -unicode-match-property-ecmascript@^1.0.4: - version "1.0.4" - resolved "https://registry.yarnpkg.com/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-1.0.4.tgz#8ed2a32569961bce9227d09cd3ffbb8fed5f020c" - integrity sha512-L4Qoh15vTfntsn4P1zqnHulG0LdXgjSO035fEpdtp6YxXhMT51Q6vgM5lYdG/5X3MjS+k/Y9Xw4SFCY9IkR0rg== - dependencies: - unicode-canonical-property-names-ecmascript "^1.0.4" - unicode-property-aliases-ecmascript "^1.0.4" - -unicode-match-property-value-ecmascript@^1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-1.2.0.tgz#0d91f600eeeb3096aa962b1d6fc88876e64ea531" - integrity sha512-wjuQHGQVofmSJv1uVISKLE5zO2rNGzM/KCYZch/QQvez7C1hUhBIuZ701fYXExuufJFMPhv2SyL8CyoIfMLbIQ== - -unicode-property-aliases-ecmascript@^1.0.4: - version "1.1.0" - resolved "https://registry.yarnpkg.com/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-1.1.0.tgz#dd57a99f6207bedff4628abefb94c50db941c8f4" - integrity sha512-PqSoPh/pWetQ2phoj5RLiaqIk4kCNwoV3CI+LfGmWLKI3rE3kl1h59XpX2BjgDrmbxD9ARtQobPGU1SguCYuQg== - -union-value@^1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/union-value/-/union-value-1.0.1.tgz#0b6fe7b835aecda61c6ea4d4f02c14221e109847" - integrity sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg== - dependencies: - arr-union "^3.1.0" - get-value "^2.0.6" - is-extendable "^0.1.1" - set-value "^2.0.1" - -uniq@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/uniq/-/uniq-1.0.1.tgz#b31c5ae8254844a3a8281541ce2b04b865a734ff" - integrity sha1-sxxa6CVIRKOoKBVBzisEuGWnNP8= - -uniqs@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/uniqs/-/uniqs-2.0.0.tgz#ffede4b36b25290696e6e165d4a59edb998e6b02" - integrity sha1-/+3ks2slKQaW5uFl1KWe25mOawI= - -unique-filename@^1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/unique-filename/-/unique-filename-1.1.1.tgz#1d69769369ada0583103a1e6ae87681b56573230" - integrity sha512-Vmp0jIp2ln35UTXuryvjzkjGdRyf9b2lTXuSYUiPmzRcl3FDtYqAwOnTJkAngD9SWhnoJzDbTKwaOrZ+STtxNQ== - dependencies: - unique-slug "^2.0.0" - -unique-slug@^2.0.0: - version "2.0.2" - resolved "https://registry.yarnpkg.com/unique-slug/-/unique-slug-2.0.2.tgz#baabce91083fc64e945b0f3ad613e264f7cd4e6c" - integrity sha512-zoWr9ObaxALD3DOPfjPSqxt4fnZiWblxHIgeWqW8x7UqDzEtHEQLzji2cuJYQFCU6KmoJikOYAZlrTHHebjx2w== - dependencies: - imurmurhash "^0.1.4" - -universal-analytics@0.4.23: - version "0.4.23" - resolved "https://registry.yarnpkg.com/universal-analytics/-/universal-analytics-0.4.23.tgz#d915e676850c25c4156762471bdd7cf2eaaca8ac" - integrity sha512-lgMIH7XBI6OgYn1woDEmxhGdj8yDefMKg7GkWdeATAlQZFrMrNyxSkpDzY57iY0/6fdlzTbBV03OawvvzG+q7A== - dependencies: - debug "^4.1.1" - request "^2.88.2" - uuid "^3.0.0" - -universalify@^0.1.0: - version "0.1.2" - resolved "https://registry.yarnpkg.com/universalify/-/universalify-0.1.2.tgz#b646f69be3942dabcecc9d6639c80dc105efaa66" - integrity sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg== - -unpipe@1.0.0, unpipe@~1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/unpipe/-/unpipe-1.0.0.tgz#b2bf4ee8514aae6165b4817829d21b2ef49904ec" - integrity sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw= - -unquote@~1.1.1: - version "1.1.1" - resolved "https://registry.yarnpkg.com/unquote/-/unquote-1.1.1.tgz#8fded7324ec6e88a0ff8b905e7c098cdc086d544" - integrity sha1-j97XMk7G6IoP+LkF58CYzcCG1UQ= - -unset-value@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/unset-value/-/unset-value-1.0.0.tgz#8376873f7d2335179ffb1e6fc3a8ed0dfc8ab559" - integrity sha1-g3aHP30jNRef+x5vw6jtDfyKtVk= - dependencies: - has-value "^0.3.1" - isobject "^3.0.0" - -upath@^1.1.1: - version "1.2.0" - resolved "https://registry.yarnpkg.com/upath/-/upath-1.2.0.tgz#8f66dbcd55a883acdae4408af8b035a5044c1894" - integrity sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg== - -uri-js@^4.2.2: - version "4.4.1" - resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" - integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg== - dependencies: - punycode "^2.1.0" - -urix@^0.1.0: - version "0.1.0" - resolved "https://registry.yarnpkg.com/urix/-/urix-0.1.0.tgz#da937f7a62e21fec1fd18d49b35c2935067a6c72" - integrity sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI= - -url-parse-lax@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/url-parse-lax/-/url-parse-lax-3.0.0.tgz#16b5cafc07dbe3676c1b1999177823d6503acb0c" - integrity sha1-FrXK/Afb42dsGxmZF3gj1lA6yww= - dependencies: - prepend-http "^2.0.0" - -url-parse@^1.4.3: - version "1.5.1" - resolved "https://registry.yarnpkg.com/url-parse/-/url-parse-1.5.1.tgz#d5fa9890af8a5e1f274a2c98376510f6425f6e3b" - integrity sha512-HOfCOUJt7iSYzEx/UqgtwKRMC6EU91NFhsCHMv9oM03VJcVo2Qrp8T8kI9D7amFf1cu+/3CEhgb3rF9zL7k85Q== - dependencies: - querystringify "^2.1.1" - requires-port "^1.0.0" - -url@^0.11.0: - version "0.11.0" - resolved "https://registry.yarnpkg.com/url/-/url-0.11.0.tgz#3838e97cfc60521eb73c525a8e55bfdd9e2e28f1" - integrity sha1-ODjpfPxgUh63PFJajlW/3Z4uKPE= - dependencies: - punycode "1.3.2" - querystring "0.2.0" - -use@^3.1.0: - version "3.1.1" - resolved "https://registry.yarnpkg.com/use/-/use-3.1.1.tgz#d50c8cac79a19fbc20f2911f56eb973f4e10070f" - integrity sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ== - -util-deprecate@^1.0.1, util-deprecate@^1.0.2, util-deprecate@~1.0.1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" - integrity sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8= - -util.promisify@~1.0.0: - version "1.0.1" - resolved "https://registry.yarnpkg.com/util.promisify/-/util.promisify-1.0.1.tgz#6baf7774b80eeb0f7520d8b81d07982a59abbaee" - integrity sha512-g9JpC/3He3bm38zsLupWryXHoEcS22YHthuPQSJdMy6KNrzIRzWqcsHzD/WUnqe45whVou4VIsPew37DoXWNrA== - dependencies: - define-properties "^1.1.3" - es-abstract "^1.17.2" - has-symbols "^1.0.1" - object.getownpropertydescriptors "^2.1.0" - -util@0.10.3: - version "0.10.3" - resolved "https://registry.yarnpkg.com/util/-/util-0.10.3.tgz#7afb1afe50805246489e3db7fe0ed379336ac0f9" - integrity sha1-evsa/lCAUkZInj23/g7TeTNqwPk= - dependencies: - inherits "2.0.1" - -util@^0.11.0: - version "0.11.1" - resolved "https://registry.yarnpkg.com/util/-/util-0.11.1.tgz#3236733720ec64bb27f6e26f421aaa2e1b588d61" - integrity sha512-HShAsny+zS2TZfaXxD9tYj4HQGlBezXZMZuM/S5PKLLoZkShZiGk9o5CzukI1LVHZvjdvZ2Sj1aW/Ndn2NB/HQ== - dependencies: - inherits "2.0.3" - -utils-merge@1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713" - integrity sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM= - -uuid@8.3.1: - version "8.3.1" - resolved "https://registry.yarnpkg.com/uuid/-/uuid-8.3.1.tgz#2ba2e6ca000da60fce5a196954ab241131e05a31" - integrity sha512-FOmRr+FmWEIG8uhZv6C2bTgEVXsHk08kE7mPlrBbEe+c3r9pjceVPgupIfNIhc4yx55H69OXANrUaSuu9eInKg== - -uuid@^3.0.0, uuid@^3.3.2, uuid@^3.4.0: - version "3.4.0" - resolved "https://registry.yarnpkg.com/uuid/-/uuid-3.4.0.tgz#b23e4358afa8a202fe7a100af1f5f883f02007ee" - integrity sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A== - -validate-npm-package-license@^3.0.1: - version "3.0.4" - resolved "https://registry.yarnpkg.com/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz#fc91f6b9c7ba15c857f4cb2c5defeec39d4f410a" - integrity sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew== - dependencies: - spdx-correct "^3.0.0" - spdx-expression-parse "^3.0.0" - -validate-npm-package-name@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/validate-npm-package-name/-/validate-npm-package-name-3.0.0.tgz#5fa912d81eb7d0c74afc140de7317f0ca7df437e" - integrity sha1-X6kS2B630MdK/BQN5zF/DKffQ34= - dependencies: - builtins "^1.0.3" - -vary@~1.1.2: - version "1.1.2" - resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc" - integrity sha1-IpnwLG3tMNSllhsLn3RSShj2NPw= - -vendors@^1.0.0: - version "1.0.4" - resolved "https://registry.yarnpkg.com/vendors/-/vendors-1.0.4.tgz#e2b800a53e7a29b93506c3cf41100d16c4c4ad8e" - integrity sha512-/juG65kTL4Cy2su4P8HjtkTxk6VmJDiOPBufWniqQ6wknac6jNiXS9vU+hO3wgusiyqWlzTbVHi0dyJqRONg3w== - -verror@1.10.0: - version "1.10.0" - resolved "https://registry.yarnpkg.com/verror/-/verror-1.10.0.tgz#3a105ca17053af55d6e270c1f8288682e18da400" - integrity sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA= - dependencies: - assert-plus "^1.0.0" - core-util-is "1.0.2" - extsprintf "^1.2.0" - -vm-browserify@^1.0.1: - version "1.1.2" - resolved "https://registry.yarnpkg.com/vm-browserify/-/vm-browserify-1.1.2.tgz#78641c488b8e6ca91a75f511e7a3b32a86e5dda0" - integrity sha512-2ham8XPWTONajOR0ohOKOHXkm3+gaBmGut3SRuu75xLd/RRaY6vqgh8NBYYk7+RW3u5AtzPQZG8F10LHkl0lAQ== - -void-elements@^2.0.0: - version "2.0.1" - resolved "https://registry.yarnpkg.com/void-elements/-/void-elements-2.0.1.tgz#c066afb582bb1cb4128d60ea92392e94d5e9dbec" - integrity sha1-wGavtYK7HLQSjWDqkjkulNXp2+w= - -watchpack-chokidar2@^2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/watchpack-chokidar2/-/watchpack-chokidar2-2.0.1.tgz#38500072ee6ece66f3769936950ea1771be1c957" - integrity sha512-nCFfBIPKr5Sh61s4LPpy1Wtfi0HE8isJ3d2Yb5/Ppw2P2B/3eVSEBjKfN0fmHJSK14+31KwMKmcrzs2GM4P0Ww== - dependencies: - chokidar "^2.1.8" - -watchpack@^1.7.4: - version "1.7.5" - resolved "https://registry.yarnpkg.com/watchpack/-/watchpack-1.7.5.tgz#1267e6c55e0b9b5be44c2023aed5437a2c26c453" - integrity sha512-9P3MWk6SrKjHsGkLT2KHXdQ/9SNkyoJbabxnKOoJepsvJjJG8uYTR3yTPxPQvNDI3w4Nz1xnE0TLHK4RIVe/MQ== - dependencies: - graceful-fs "^4.1.2" - neo-async "^2.5.0" - optionalDependencies: - chokidar "^3.4.1" - watchpack-chokidar2 "^2.0.1" - -wbuf@^1.1.0, wbuf@^1.7.3: - version "1.7.3" - resolved "https://registry.yarnpkg.com/wbuf/-/wbuf-1.7.3.tgz#c1d8d149316d3ea852848895cb6a0bfe887b87df" - integrity sha512-O84QOnr0icsbFGLS0O3bI5FswxzRr8/gHwWkDlQFskhSPryQXvrTMxjxGP4+iWYoauLoBvfDpkrOauZ+0iZpDA== - dependencies: - minimalistic-assert "^1.0.0" - -wcwidth@^1.0.1: - version "1.0.1" - resolved "https://registry.yarnpkg.com/wcwidth/-/wcwidth-1.0.1.tgz#f0b0dcf915bc5ff1528afadb2c0e17b532da2fe8" - integrity sha1-8LDc+RW8X/FSivrbLA4XtTLaL+g= - dependencies: - defaults "^1.0.3" - -webdriver-js-extender@2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/webdriver-js-extender/-/webdriver-js-extender-2.1.0.tgz#57d7a93c00db4cc8d556e4d3db4b5db0a80c3bb7" - integrity sha512-lcUKrjbBfCK6MNsh7xaY2UAUmZwe+/ib03AjVOpFobX4O7+83BUveSrLfU0Qsyb1DaKJdQRbuU+kM9aZ6QUhiQ== - dependencies: - "@types/selenium-webdriver" "^3.0.0" - selenium-webdriver "^3.0.1" - -webdriver-manager@^12.1.7: - version "12.1.8" - resolved "https://registry.yarnpkg.com/webdriver-manager/-/webdriver-manager-12.1.8.tgz#5e70e73eaaf53a0767d5745270addafbc5905fd4" - integrity sha512-qJR36SXG2VwKugPcdwhaqcLQOD7r8P2Xiv9sfNbfZrKBnX243iAkOueX1yAmeNgIKhJ3YAT/F2gq6IiEZzahsg== - dependencies: - adm-zip "^0.4.9" - chalk "^1.1.1" - del "^2.2.0" - glob "^7.0.3" - ini "^1.3.4" - minimist "^1.2.0" - q "^1.4.1" - request "^2.87.0" - rimraf "^2.5.2" - semver "^5.3.0" - xml2js "^0.4.17" - -webpack-dev-middleware@3.7.2: - version "3.7.2" - resolved "https://registry.yarnpkg.com/webpack-dev-middleware/-/webpack-dev-middleware-3.7.2.tgz#0019c3db716e3fa5cecbf64f2ab88a74bab331f3" - integrity sha512-1xC42LxbYoqLNAhV6YzTYacicgMZQTqRd27Sim9wn5hJrX3I5nxYy1SxSd4+gjUFsz1dQFj+yEe6zEVmSkeJjw== - dependencies: - memory-fs "^0.4.1" - mime "^2.4.4" - mkdirp "^0.5.1" - range-parser "^1.2.1" - webpack-log "^2.0.0" - -webpack-dev-middleware@^3.7.2: - version "3.7.3" - resolved "https://registry.yarnpkg.com/webpack-dev-middleware/-/webpack-dev-middleware-3.7.3.tgz#0639372b143262e2b84ab95d3b91a7597061c2c5" - integrity sha512-djelc/zGiz9nZj/U7PTBi2ViorGJXEWo/3ltkPbDyxCXhhEXkW0ce99falaok4TPj+AsxLiXJR0EBOb0zh9fKQ== - dependencies: - memory-fs "^0.4.1" - mime "^2.4.4" - mkdirp "^0.5.1" - range-parser "^1.2.1" - webpack-log "^2.0.0" - -webpack-dev-server@3.11.0: - version "3.11.0" - resolved "https://registry.yarnpkg.com/webpack-dev-server/-/webpack-dev-server-3.11.0.tgz#8f154a3bce1bcfd1cc618ef4e703278855e7ff8c" - integrity sha512-PUxZ+oSTxogFQgkTtFndEtJIPNmml7ExwufBZ9L2/Xyyd5PnOL5UreWe5ZT7IU25DSdykL9p1MLQzmLh2ljSeg== - dependencies: - ansi-html "0.0.7" - bonjour "^3.5.0" - chokidar "^2.1.8" - compression "^1.7.4" - connect-history-api-fallback "^1.6.0" - debug "^4.1.1" - del "^4.1.1" - express "^4.17.1" - html-entities "^1.3.1" - http-proxy-middleware "0.19.1" - import-local "^2.0.0" - internal-ip "^4.3.0" - ip "^1.1.5" - is-absolute-url "^3.0.3" - killable "^1.0.1" - loglevel "^1.6.8" - opn "^5.5.0" - p-retry "^3.0.1" - portfinder "^1.0.26" - schema-utils "^1.0.0" - selfsigned "^1.10.7" - semver "^6.3.0" - serve-index "^1.9.1" - sockjs "0.3.20" - sockjs-client "1.4.0" - spdy "^4.0.2" - strip-ansi "^3.0.1" - supports-color "^6.1.0" - url "^0.11.0" - webpack-dev-middleware "^3.7.2" - webpack-log "^2.0.0" - ws "^6.2.1" - yargs "^13.3.2" - -webpack-log@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/webpack-log/-/webpack-log-2.0.0.tgz#5b7928e0637593f119d32f6227c1e0ac31e1b47f" - integrity sha512-cX8G2vR/85UYG59FgkoMamwHUIkSSlV3bBMRsbxVXVUk2j6NleCKjQ/WE9eYg9WY4w25O9w8wKP4rzNZFmUcUg== - dependencies: - ansi-colors "^3.0.0" - uuid "^3.3.2" - -webpack-merge@5.2.0: - version "5.2.0" - resolved "https://registry.yarnpkg.com/webpack-merge/-/webpack-merge-5.2.0.tgz#31cbcc954f8f89cd4b06ca8d97a38549f7f3f0c9" - integrity sha512-QBglJBg5+lItm3/Lopv8KDDK01+hjdg2azEwi/4vKJ8ZmGPdtJsTpjtNNOW3a4WiqzXdCATtTudOZJngE7RKkA== - dependencies: - clone-deep "^4.0.1" - wildcard "^2.0.0" - -webpack-sources@2.0.1: - version "2.0.1" - resolved "https://registry.yarnpkg.com/webpack-sources/-/webpack-sources-2.0.1.tgz#1467f6e692ddce91e88b8044c44347b1087bbd4f" - integrity sha512-A9oYz7ANQBK5EN19rUXbvNgfdfZf5U2gP0769OXsj9CvYkCR6OHOsd6OKyEy4H38GGxpsQPKIL83NC64QY6Xmw== - dependencies: - source-list-map "^2.0.1" - source-map "^0.6.1" - -webpack-sources@^1.1.0, webpack-sources@^1.2.0, webpack-sources@^1.3.0, webpack-sources@^1.4.0, webpack-sources@^1.4.1, webpack-sources@^1.4.3: - version "1.4.3" - resolved "https://registry.yarnpkg.com/webpack-sources/-/webpack-sources-1.4.3.tgz#eedd8ec0b928fbf1cbfe994e22d2d890f330a933" - integrity sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ== - dependencies: - source-list-map "^2.0.0" - source-map "~0.6.1" - -webpack-subresource-integrity@1.5.1: - version "1.5.1" - resolved "https://registry.yarnpkg.com/webpack-subresource-integrity/-/webpack-subresource-integrity-1.5.1.tgz#6f44ea99987266b70c4ec42ac51064d33e982277" - integrity sha512-uekbQ93PZ9e7BFB8Hl9cFIVYQyQqiXp2ExKk9Zv+qZfH/zHXHrCFAfw1VW0+NqWbTWrs/HnuDrto3+tiPXh//Q== - dependencies: - webpack-sources "^1.3.0" - -webpack@4.44.2: - version "4.44.2" - resolved "https://registry.yarnpkg.com/webpack/-/webpack-4.44.2.tgz#6bfe2b0af055c8b2d1e90ed2cd9363f841266b72" - integrity sha512-6KJVGlCxYdISyurpQ0IPTklv+DULv05rs2hseIXer6D7KrUicRDLFb4IUM1S6LUAKypPM/nSiVSuv8jHu1m3/Q== - dependencies: - "@webassemblyjs/ast" "1.9.0" - "@webassemblyjs/helper-module-context" "1.9.0" - "@webassemblyjs/wasm-edit" "1.9.0" - "@webassemblyjs/wasm-parser" "1.9.0" - acorn "^6.4.1" - ajv "^6.10.2" - ajv-keywords "^3.4.1" - chrome-trace-event "^1.0.2" - enhanced-resolve "^4.3.0" - eslint-scope "^4.0.3" - json-parse-better-errors "^1.0.2" - loader-runner "^2.4.0" - loader-utils "^1.2.3" - memory-fs "^0.4.1" - micromatch "^3.1.10" - mkdirp "^0.5.3" - neo-async "^2.6.1" - node-libs-browser "^2.2.1" - schema-utils "^1.0.0" - tapable "^1.1.3" - terser-webpack-plugin "^1.4.3" - watchpack "^1.7.4" - webpack-sources "^1.4.1" - -websocket-driver@0.6.5: - version "0.6.5" - resolved "https://registry.yarnpkg.com/websocket-driver/-/websocket-driver-0.6.5.tgz#5cb2556ceb85f4373c6d8238aa691c8454e13a36" - integrity sha1-XLJVbOuF9Dc8bYI4qmkchFThOjY= - dependencies: - websocket-extensions ">=0.1.1" - -websocket-driver@>=0.5.1: - version "0.7.4" - resolved "https://registry.yarnpkg.com/websocket-driver/-/websocket-driver-0.7.4.tgz#89ad5295bbf64b480abcba31e4953aca706f5760" - integrity sha512-b17KeDIQVjvb0ssuSDF2cYXSg2iztliJ4B9WdsuB6J952qCPKmnVq4DyW5motImXHDC1cBT/1UezrJVsKw5zjg== - dependencies: - http-parser-js ">=0.5.1" - safe-buffer ">=5.1.0" - websocket-extensions ">=0.1.1" - -websocket-extensions@>=0.1.1: - version "0.1.4" - resolved "https://registry.yarnpkg.com/websocket-extensions/-/websocket-extensions-0.1.4.tgz#7f8473bc839dfd87608adb95d7eb075211578a42" - integrity sha512-OqedPIGOfsDlo31UNwYbCFMSaO9m9G/0faIHj5/dZFDMFqPTcx6UwqyOy3COEaEOg/9VsGIpdqn62W5KhoKSpg== - -whatwg-mimetype@^2.3.0: - version "2.3.0" - resolved "https://registry.yarnpkg.com/whatwg-mimetype/-/whatwg-mimetype-2.3.0.tgz#3d4b1e0312d2079879f826aff18dbeeca5960fbf" - integrity sha512-M4yMwr6mAnQz76TbJm914+gPpB/nCwvZbJU28cUD6dR004SAxDLOOSUaB1JDRqLtaOV/vi0IC5lEAGFgrjGv/g== - -which-boxed-primitive@^1.0.1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz#13757bc89b209b049fe5d86430e21cf40a89a8e6" - integrity sha512-bwZdv0AKLpplFY2KZRX6TvyuN7ojjr7lwkg6ml0roIy9YeuSr7JS372qlNW18UQYzgYK9ziGcerWqZOmEn9VNg== - dependencies: - is-bigint "^1.0.1" - is-boolean-object "^1.1.0" - is-number-object "^1.0.4" - is-string "^1.0.5" - is-symbol "^1.0.3" - -which-module@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/which-module/-/which-module-2.0.0.tgz#d9ef07dce77b9902b8a3a8fa4b31c3e3f7e6e87a" - integrity sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho= - -which@^1.2.1, which@^1.2.9, which@^1.3.1: - version "1.3.1" - resolved "https://registry.yarnpkg.com/which/-/which-1.3.1.tgz#a45043d54f5805316da8d62f9f50918d3da70b0a" - integrity sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ== - dependencies: - isexe "^2.0.0" - -wildcard@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/wildcard/-/wildcard-2.0.0.tgz#a77d20e5200c6faaac979e4b3aadc7b3dd7f8fec" - integrity sha512-JcKqAHLPxcdb9KM49dufGXn2x3ssnfjbcaQdLlfZsL9rH9wgDQjUtDxbo8NE0F6SFvydeu1VhZe7hZuHsB2/pw== - -worker-farm@^1.7.0: - version "1.7.0" - resolved "https://registry.yarnpkg.com/worker-farm/-/worker-farm-1.7.0.tgz#26a94c5391bbca926152002f69b84a4bf772e5a8" - integrity sha512-rvw3QTZc8lAxyVrqcSGVm5yP/IJ2UcB3U0graE3LCFoZ0Yn2x4EoVSqJKdB/T5M+FLcRPjz4TDacRf3OCfNUzw== - dependencies: - errno "~0.1.7" - -worker-plugin@5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/worker-plugin/-/worker-plugin-5.0.0.tgz#113b5fe1f4a5d6a957cecd29915bedafd70bb537" - integrity sha512-AXMUstURCxDD6yGam2r4E34aJg6kW85IiaeX72hi+I1cxyaMUtrvVY6sbfpGKAj5e7f68Acl62BjQF5aOOx2IQ== - dependencies: - loader-utils "^1.1.0" - -wrap-ansi@^5.1.0: - version "5.1.0" - resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-5.1.0.tgz#1fd1f67235d5b6d0fee781056001bfb694c03b09" - integrity sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q== - dependencies: - ansi-styles "^3.2.0" - string-width "^3.0.0" - strip-ansi "^5.0.0" - -wrap-ansi@^6.2.0: - version "6.2.0" - resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-6.2.0.tgz#e9393ba07102e6c91a3b221478f0257cd2856e53" - integrity sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA== - dependencies: - ansi-styles "^4.0.0" - string-width "^4.1.0" - strip-ansi "^6.0.0" - -wrap-ansi@^7.0.0: - version "7.0.0" - resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" - integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== - dependencies: - ansi-styles "^4.0.0" - string-width "^4.1.0" - strip-ansi "^6.0.0" - -wrappy@1: - version "1.0.2" - resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" - integrity sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8= - -ws@^6.2.1: - version "6.2.1" - resolved "https://registry.yarnpkg.com/ws/-/ws-6.2.1.tgz#442fdf0a47ed64f59b6a5d8ff130f4748ed524fb" - integrity sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA== - dependencies: - async-limiter "~1.0.0" - -ws@~7.4.2: - version "7.4.4" - resolved "https://registry.yarnpkg.com/ws/-/ws-7.4.4.tgz#383bc9742cb202292c9077ceab6f6047b17f2d59" - integrity sha512-Qm8k8ojNQIMx7S+Zp8u/uHOx7Qazv3Yv4q68MiWWWOJhiwG5W3x7iqmRtJo8xxrciZUY4vRxUTJCKuRnF28ZZw== - -xml2js@^0.4.17: - version "0.4.23" - resolved "https://registry.yarnpkg.com/xml2js/-/xml2js-0.4.23.tgz#a0c69516752421eb2ac758ee4d4ccf58843eac66" - integrity sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug== - dependencies: - sax ">=0.6.0" - xmlbuilder "~11.0.0" - -xmlbuilder@~11.0.0: - version "11.0.1" - resolved "https://registry.yarnpkg.com/xmlbuilder/-/xmlbuilder-11.0.1.tgz#be9bae1c8a046e76b31127726347d0ad7002beb3" - integrity sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA== - -xmlhttprequest-ssl@~1.5.4: - version "1.5.5" - resolved "https://registry.yarnpkg.com/xmlhttprequest-ssl/-/xmlhttprequest-ssl-1.5.5.tgz#c2876b06168aadc40e57d97e81191ac8f4398b3e" - integrity sha1-wodrBhaKrcQOV9l+gRkayPQ5iz4= - -xtend@^4.0.0, xtend@~4.0.1: - version "4.0.2" - resolved "https://registry.yarnpkg.com/xtend/-/xtend-4.0.2.tgz#bb72779f5fa465186b1f438f674fa347fdb5db54" - integrity sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ== - -y18n@^4.0.0: - version "4.0.1" - resolved "https://registry.yarnpkg.com/y18n/-/y18n-4.0.1.tgz#8db2b83c31c5d75099bb890b23f3094891e247d4" - integrity sha512-wNcy4NvjMYL8gogWWYAO7ZFWFfHcbdbE57tZO8e4cbpj8tfUcwrwqSl3ad8HxpYWCdXcJUCeKKZS62Av1affwQ== - -y18n@^5.0.5: - version "5.0.5" - resolved "https://registry.yarnpkg.com/y18n/-/y18n-5.0.5.tgz#8769ec08d03b1ea2df2500acef561743bbb9ab18" - integrity sha512-hsRUr4FFrvhhRH12wOdfs38Gy7k2FFzB9qgN9v3aLykRq0dRcdcpz5C9FxdS2NuhOrI/628b/KSTJ3rwHysYSg== - -yallist@^3.0.0, yallist@^3.0.2, yallist@^3.0.3: - version "3.1.1" - resolved "https://registry.yarnpkg.com/yallist/-/yallist-3.1.1.tgz#dbb7daf9bfd8bac9ab45ebf602b8cbad0d5d08fd" - integrity sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g== - -yallist@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72" - integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A== - -yaml@^1.10.0: - version "1.10.1" - resolved "https://registry.yarnpkg.com/yaml/-/yaml-1.10.1.tgz#bb13d805ed104fba38f533570f3441027eeeca22" - integrity sha512-z/asvd+V08l1ywhaemZVirCwjdzLo6O1/0j2JbYCsGjiezupNQqjs5IIPyNtctbHjPEckqzVGd4jvpU5Lr25vQ== - -yargs-parser@^13.1.2: - version "13.1.2" - resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-13.1.2.tgz#130f09702ebaeef2650d54ce6e3e5706f7a4fb38" - integrity sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg== - dependencies: - camelcase "^5.0.0" - decamelize "^1.2.0" - -yargs-parser@^18.1.2: - version "18.1.3" - resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-18.1.3.tgz#be68c4975c6b2abf469236b0c870362fab09a7b0" - integrity sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ== - dependencies: - camelcase "^5.0.0" - decamelize "^1.2.0" - -yargs-parser@^20.2.2: - version "20.2.7" - resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.7.tgz#61df85c113edfb5a7a4e36eb8aa60ef423cbc90a" - integrity sha512-FiNkvbeHzB/syOjIUxFDCnhSfzAL8R5vs40MgLFBorXACCOAEaWu0gRZl14vG8MR9AOJIZbmkjhusqBYZ3HTHw== - -yargs@^13.3.2: - version "13.3.2" - resolved "https://registry.yarnpkg.com/yargs/-/yargs-13.3.2.tgz#ad7ffefec1aa59565ac915f82dccb38a9c31a2dd" - integrity sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw== - dependencies: - cliui "^5.0.0" - find-up "^3.0.0" - get-caller-file "^2.0.1" - require-directory "^2.1.1" - require-main-filename "^2.0.0" - set-blocking "^2.0.0" - string-width "^3.0.0" - which-module "^2.0.0" - y18n "^4.0.0" - yargs-parser "^13.1.2" - -yargs@^15.3.1: - version "15.4.1" - resolved "https://registry.yarnpkg.com/yargs/-/yargs-15.4.1.tgz#0d87a16de01aee9d8bec2bfbf74f67851730f4f8" - integrity sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A== - dependencies: - cliui "^6.0.0" - decamelize "^1.2.0" - find-up "^4.1.0" - get-caller-file "^2.0.1" - require-directory "^2.1.1" - require-main-filename "^2.0.0" - set-blocking "^2.0.0" - string-width "^4.2.0" - which-module "^2.0.0" - y18n "^4.0.0" - yargs-parser "^18.1.2" - -yargs@^16.1.1: - version "16.2.0" - resolved "https://registry.yarnpkg.com/yargs/-/yargs-16.2.0.tgz#1c82bf0f6b6a66eafce7ef30e376f49a12477f66" - integrity sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw== - dependencies: - cliui "^7.0.2" - escalade "^3.1.1" - get-caller-file "^2.0.5" - require-directory "^2.1.1" - string-width "^4.2.0" - y18n "^5.0.5" - yargs-parser "^20.2.2" - -yauzl@^2.10.0: - version "2.10.0" - resolved "https://registry.yarnpkg.com/yauzl/-/yauzl-2.10.0.tgz#c7eb17c93e112cb1086fa6d8e51fb0667b79a5f9" - integrity sha1-x+sXyT4RLLEIb6bY5R+wZnt5pfk= - dependencies: - buffer-crc32 "~0.2.3" - fd-slicer "~1.1.0" - -yeast@0.1.2: - version "0.1.2" - resolved "https://registry.yarnpkg.com/yeast/-/yeast-0.1.2.tgz#008e06d8094320c372dbc2f8ed76a0ca6c8ac419" - integrity sha1-AI4G2AlDIMNy28L47XagymyKxBk= - -yn@3.1.1: - version "3.1.1" - resolved "https://registry.yarnpkg.com/yn/-/yn-3.1.1.tgz#1e87401a09d767c1d5eab26a6e4c185182d2eb50" - integrity sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q== - -yocto-queue@^0.1.0: - version "0.1.0" - resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b" - integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q== - -zone.js@0.10.3, zone.js@~0.10.3: - version "0.10.3" - resolved "https://registry.yarnpkg.com/zone.js/-/zone.js-0.10.3.tgz#3e5e4da03c607c9dcd92e37dd35687a14a140c16" - integrity sha512-LXVLVEq0NNOqK/fLJo3d0kfzd4sxwn2/h67/02pjCjfKDxgx1i9QqpvtHD8CrBnSSwMw5+dy11O7FRX5mkO7Cg== diff --git a/vripper-ui/.angulardoc.json b/vripper-web-ui/.angulardoc.json similarity index 100% rename from vripper-ui/.angulardoc.json rename to vripper-web-ui/.angulardoc.json diff --git a/vripper-ui/.editorconfig b/vripper-web-ui/.editorconfig similarity index 100% rename from vripper-ui/.editorconfig rename to vripper-web-ui/.editorconfig diff --git a/vripper-ui/.gitignore b/vripper-web-ui/.gitignore similarity index 100% rename from vripper-ui/.gitignore rename to vripper-web-ui/.gitignore diff --git a/vripper-ui/.prettierrc b/vripper-web-ui/.prettierrc similarity index 100% rename from vripper-ui/.prettierrc rename to vripper-web-ui/.prettierrc diff --git a/vripper-ui/README.md b/vripper-web-ui/README.md similarity index 100% rename from vripper-ui/README.md rename to vripper-web-ui/README.md diff --git a/vripper-ui/angular.json b/vripper-web-ui/angular.json similarity index 53% rename from vripper-ui/angular.json rename to vripper-web-ui/angular.json index 348943ce..98a47f5c 100644 --- a/vripper-ui/angular.json +++ b/vripper-web-ui/angular.json @@ -17,12 +17,14 @@ "build": { "builder": "@angular-devkit/build-angular:browser", "options": { - "aot": true, "outputPath": "dist/vripper-ui", "index": "src/index.html", "main": "src/main.ts", - "polyfills": "src/polyfills.ts", - "tsConfig": "src/tsconfig.app.json", + "polyfills": [ + "zone.js" + ], + "tsConfig": "tsconfig.app.json", + "inlineStyleLanguage": "scss", "assets": [ "src/favicon.ico", "src/assets", @@ -41,20 +43,6 @@ }, "configurations": { "production": { - "fileReplacements": [ - { - "replace": "src/environments/environment.ts", - "with": "src/environments/environment.prod.ts" - } - ], - "optimization": true, - "outputHashing": "all", - "sourceMap": false, - "namedChunks": false, - "aot": true, - "extractLicenses": true, - "vendorChunk": false, - "buildOptimizer": true, "budgets": [ { "type": "initial", @@ -66,19 +54,29 @@ "maximumWarning": "6kb" } ] + }, + "development": { + "buildOptimizer": false, + "optimization": false, + "vendorChunk": true, + "extractLicenses": false, + "sourceMap": true, + "namedChunks": true } - } + }, + "defaultConfiguration": "production" }, "serve": { "builder": "@angular-devkit/build-angular:dev-server", - "options": { - "browserTarget": "vripper-ui:build" - }, "configurations": { "production": { "browserTarget": "vripper-ui:build:production" + }, + "development": { + "browserTarget": "vripper-ui:build:development" } - } + }, + "defaultConfiguration": "development" }, "extract-i18n": { "builder": "@angular-devkit/build-angular:extract-i18n", @@ -89,10 +87,12 @@ "test": { "builder": "@angular-devkit/build-angular:karma", "options": { - "main": "src/test.ts", - "polyfills": "src/polyfills.ts", - "tsConfig": "src/tsconfig.spec.json", - "karmaConfig": "src/karma.conf.js", + "polyfills": [ + "zone.js", + "zone.js/testing" + ], + "tsConfig": "tsconfig.spec.json", + "inlineStyleLanguage": "scss", "styles": [ "src/styles.scss" ], @@ -100,51 +100,11 @@ "src/favicon.ico", "src/assets", { "glob": "mdi.svg", "input": "./node_modules/@mdi/angular-material", "output": "./assets" } - ] - } - }, - "lint": { - "builder": "@angular-devkit/build-angular:tslint", - "options": { - "tsConfig": [ - "src/tsconfig.app.json", - "src/tsconfig.spec.json" ], - "exclude": [ - "**/node_modules/**" - ] - } - } - } - }, - "vripper-ui-e2e": { - "root": "e2e/", - "projectType": "application", - "prefix": "", - "architect": { - "e2e": { - "builder": "@angular-devkit/build-angular:protractor", - "options": { - "protractorConfig": "e2e/protractor.conf.js", - "devServerTarget": "vripper-ui:serve" - }, - "configurations": { - "production": { - "devServerTarget": "vripper-ui:serve:production" - } - } - }, - "lint": { - "builder": "@angular-devkit/build-angular:tslint", - "options": { - "tsConfig": "e2e/tsconfig.e2e.json", - "exclude": [ - "**/node_modules/**" - ] + "scripts": [] } } } } - }, - "defaultProject": "vripper-ui" + } } diff --git a/vripper-web-ui/package.json b/vripper-web-ui/package.json new file mode 100644 index 00000000..281025dc --- /dev/null +++ b/vripper-web-ui/package.json @@ -0,0 +1,47 @@ +{ + "name": "vripper-ui", + "version": "3.5.4", + "scripts": { + "ng": "ng", + "start": "ng serve", + "build": "ng build", + "watch": "ng build --watch --configuration development", + "test": "ng test" + }, + "private": true, + "dependencies": { + "@angular/animations": "^15.2.0", + "@angular/cdk": "^15.2.0", + "@angular/common": "^15.2.0", + "@angular/compiler": "^15.2.0", + "@angular/core": "^15.2.0", + "@angular/flex-layout": "15.0.0-beta.42", + "@angular/forms": "^15.2.0", + "@angular/material": "^15.2.0", + "@angular/platform-browser": "^15.2.0", + "@angular/platform-browser-dynamic": "^15.2.0", + "@angular/router": "^15.2.0", + "@mdi/angular-material": "^7.2.96", + "@stomp/rx-stomp": "^2.0.0", + "@stomp/stompjs": "^7.0.0", + "uuid": "^9.0.0", + "ag-grid-angular": "^29.1.0", + "ag-grid-community": "^29.1.0", + "rxjs": "~7.8.0", + "tslib": "^2.3.0", + "zone.js": "~0.12.0" + }, + "devDependencies": { + "@angular-devkit/build-angular": "^15.2.4", + "@angular/cli": "~15.2.4", + "@angular/compiler-cli": "^15.2.0", + "@types/jasmine": "~4.3.0", + "jasmine-core": "~4.5.0", + "karma": "~6.4.0", + "karma-chrome-launcher": "~3.1.0", + "karma-coverage": "~2.2.0", + "karma-jasmine": "~5.1.0", + "karma-jasmine-html-reporter": "~2.0.0", + "typescript": "~4.9.4" + } +} diff --git a/vripper-ui/pom.xml b/vripper-web-ui/pom.xml similarity index 99% rename from vripper-ui/pom.xml rename to vripper-web-ui/pom.xml index f6aa4cb4..ae5906be 100644 --- a/vripper-ui/pom.xml +++ b/vripper-web-ui/pom.xml @@ -3,7 +3,7 @@ xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"> 4.0.0 - tn.mnlr + me.mnlr vripper 3.5.4 diff --git a/vripper-ui/src/.browserslistrc b/vripper-web-ui/src/.browserslistrc similarity index 100% rename from vripper-ui/src/.browserslistrc rename to vripper-web-ui/src/.browserslistrc diff --git a/vripper-web-ui/src/app/about/about-component.component-scss-theme.scss b/vripper-web-ui/src/app/about/about-component.component-scss-theme.scss new file mode 100644 index 00000000..fbdfef74 --- /dev/null +++ b/vripper-web-ui/src/app/about/about-component.component-scss-theme.scss @@ -0,0 +1,11 @@ +@use 'sass:map'; +@use '@angular/material' as mat; + +@mixin about-component-theme($theme) { + $theme-color: mat.get-color-config($theme); + $primary: map.get($theme-color, 'primary'); + + .about-link, .about-icon { + color: mat.get-color-from-palette($primary, '900-contrast'); + } +} diff --git a/vripper-ui/src/app/about/about-component.html b/vripper-web-ui/src/app/about/about-component.html similarity index 100% rename from vripper-ui/src/app/about/about-component.html rename to vripper-web-ui/src/app/about/about-component.html diff --git a/vripper-ui/src/app/about/about-component.scss b/vripper-web-ui/src/app/about/about-component.scss similarity index 100% rename from vripper-ui/src/app/about/about-component.scss rename to vripper-web-ui/src/app/about/about-component.scss diff --git a/vripper-ui/src/app/about/about-component.ts b/vripper-web-ui/src/app/about/about-component.ts similarity index 59% rename from vripper-ui/src/app/about/about-component.ts rename to vripper-web-ui/src/app/about/about-component.ts index 2f406409..08cd3a9e 100644 --- a/vripper-ui/src/app/about/about-component.ts +++ b/vripper-web-ui/src/app/about/about-component.ts @@ -1,6 +1,5 @@ import {ChangeDetectionStrategy, Component} from '@angular/core'; import {environment} from '../../environments/environment'; -import {ElectronService} from 'ngx-electron'; @Component({ selector: 'app-about', @@ -12,16 +11,7 @@ export class AboutComponent { version = environment.version; - constructor( - public electronService: ElectronService - ) { - } - goTo(url: string) { - if (this.electronService.isElectronApp) { - this.electronService.shell.openExternal(url); - } else { - window.open(url, '_blank'); - } + window.open(url, '_blank'); } } diff --git a/vripper-ui/src/app/app-routing.module.ts b/vripper-web-ui/src/app/app-routing.module.ts similarity index 85% rename from vripper-ui/src/app/app-routing.module.ts rename to vripper-web-ui/src/app/app-routing.module.ts index 78530df8..8c95cdf1 100644 --- a/vripper-ui/src/app/app-routing.module.ts +++ b/vripper-web-ui/src/app/app-routing.module.ts @@ -10,7 +10,7 @@ const routes: Routes = [ @NgModule({ imports: [ - RouterModule.forRoot(routes, {relativeLinkResolution: 'legacy'}) + RouterModule.forRoot(routes) ], exports: [ RouterModule diff --git a/vripper-ui/src/app/app.component.html b/vripper-web-ui/src/app/app.component.html similarity index 86% rename from vripper-ui/src/app/app.component.html rename to vripper-web-ui/src/app/app.component.html index 8d851690..ea7e939e 100644 --- a/vripper-ui/src/app/app.component.html +++ b/vripper-web-ui/src/app/app.component.html @@ -1,4 +1,4 @@ -
diff --git a/vripper-ui/src/app/app.component.scss b/vripper-web-ui/src/app/app.component.scss similarity index 96% rename from vripper-ui/src/app/app.component.scss rename to vripper-web-ui/src/app/app.component.scss index 6a67985f..20c6a8a7 100644 --- a/vripper-ui/src/app/app.component.scss +++ b/vripper-web-ui/src/app/app.component.scss @@ -1,3 +1,9 @@ +:host { + box-sizing: border-box; + display: block; + height: 100%; +} + .overlay-icon { font-size: 64px; width: 64px; diff --git a/vripper-web-ui/src/app/app.component.scss-theme.scss b/vripper-web-ui/src/app/app.component.scss-theme.scss new file mode 100644 index 00000000..81229175 --- /dev/null +++ b/vripper-web-ui/src/app/app.component.scss-theme.scss @@ -0,0 +1,11 @@ +@use 'sass:map'; +@use '@angular/material' as mat; + +@mixin app-component-theme($theme) { + $theme-color: mat.get-color-config($theme); + $primary: map.get($theme-color, 'primary'); + + button.add-button { + background-color: mat.get-color-from-palette($primary, 900); + } +} diff --git a/vripper-ui/src/app/app.component.ts b/vripper-web-ui/src/app/app.component.ts similarity index 82% rename from vripper-ui/src/app/app.component.ts rename to vripper-web-ui/src/app/app.component.ts index 0c8fa65a..31994329 100644 --- a/vripper-ui/src/app/app.component.ts +++ b/vripper-web-ui/src/app/app.component.ts @@ -1,6 +1,4 @@ import {AppService} from './services/app.service'; -import {ClipboardService} from './services/clipboard.service'; -import {ElectronService} from 'ngx-electron'; import {AfterViewInit, ChangeDetectionStrategy, Component, NgZone, OnDestroy, Renderer2} from '@angular/core'; import {MatDialog} from '@angular/material/dialog'; import {BehaviorSubject, merge, Subject, Subscription} from 'rxjs'; @@ -14,21 +12,16 @@ import {RxStompState} from '@stomp/rx-stomp'; changeDetection: ChangeDetectionStrategy.OnPush }) export class AppComponent implements OnDestroy, AfterViewInit { - electron: Subject; loaded: Subject = new BehaviorSubject(false); private subscriptions: Subscription[] = []; constructor( private dialog: MatDialog, public ws: WsConnectionService, - public electronService: ElectronService, - private clipboardService: ClipboardService, private appService: AppService, private renderer: Renderer2, private ngZone: NgZone - ) { - this.electron = new BehaviorSubject(electronService.isElectronApp); - } + ) {} ngAfterViewInit() { this.appService.renderer = this.renderer; @@ -36,7 +29,6 @@ export class AppComponent implements OnDestroy, AfterViewInit { this.ws.state.subscribe(state => { if (state === RxStompState.OPEN) { this.ngZone.run(() => this.loaded.next(true)); - this.clipboardService.init(); this.subscriptions.push(merge(this.appService.loadTheme(), this.appService.loadSettings()).subscribe()); } else { this.ngZone.run(() => this.loaded.next(false)); diff --git a/vripper-ui/src/app/app.module.ts b/vripper-web-ui/src/app/app.module.ts similarity index 92% rename from vripper-ui/src/app/app.module.ts rename to vripper-web-ui/src/app/app.module.ts index 8b2a0bd8..d317046a 100644 --- a/vripper-ui/src/app/app.module.ts +++ b/vripper-web-ui/src/app/app.module.ts @@ -23,8 +23,6 @@ import {ScanComponent} from './scan/scan.component'; import {StatusBarComponent} from './status-bar/status-bar.component'; import {ToolbarComponent} from './toolbar/toolbar.component'; import {MultiPostGridComponent} from './multi-post-grid/multi-post-grid.component'; -import {AlternativeTitleComponent} from './posts/alternative-title/alternative-title.component'; -import {NgxElectronModule} from 'ngx-electron'; import {MatIconRegistry} from '@angular/material/icon'; import {DomSanitizer} from '@angular/platform-browser'; import {EventLogComponent} from './event-log/event-log.component'; @@ -47,7 +45,6 @@ import {AboutComponent} from './about/about-component'; ToolbarComponent, MultiPostGridComponent, PostContextMenuComponent, - AlternativeTitleComponent, EventLogComponent, EventLogMessageDialogComponent, AboutComponent @@ -61,8 +58,7 @@ import {AboutComponent} from './about/about-component'; AppRoutingModule, ReactiveFormsModule, AgGridModule, - OverlayModule, - NgxElectronModule + OverlayModule ], bootstrap: [AppComponent] }) diff --git a/vripper-ui/src/app/confirmation-component/confirmation-dialog.html b/vripper-web-ui/src/app/confirmation-component/confirmation-dialog.html similarity index 100% rename from vripper-ui/src/app/confirmation-component/confirmation-dialog.html rename to vripper-web-ui/src/app/confirmation-component/confirmation-dialog.html diff --git a/vripper-ui/src/app/confirmation-component/confirmation-dialog.ts b/vripper-web-ui/src/app/confirmation-component/confirmation-dialog.ts similarity index 100% rename from vripper-ui/src/app/confirmation-component/confirmation-dialog.ts rename to vripper-web-ui/src/app/confirmation-component/confirmation-dialog.ts diff --git a/vripper-ui/src/app/domain/credential.model.ts b/vripper-web-ui/src/app/domain/credential.model.ts similarity index 100% rename from vripper-ui/src/app/domain/credential.model.ts rename to vripper-web-ui/src/app/domain/credential.model.ts diff --git a/vripper-ui/src/app/domain/download-path.model.ts b/vripper-web-ui/src/app/domain/download-path.model.ts similarity index 100% rename from vripper-ui/src/app/domain/download-path.model.ts rename to vripper-web-ui/src/app/domain/download-path.model.ts diff --git a/vripper-ui/src/app/domain/event.model.ts b/vripper-web-ui/src/app/domain/event.model.ts similarity index 100% rename from vripper-ui/src/app/domain/event.model.ts rename to vripper-web-ui/src/app/domain/event.model.ts diff --git a/vripper-ui/src/app/domain/global-state.model.ts b/vripper-web-ui/src/app/domain/global-state.model.ts similarity index 54% rename from vripper-ui/src/app/domain/global-state.model.ts rename to vripper-web-ui/src/app/domain/global-state.model.ts index fa0e8c74..841981e4 100644 --- a/vripper-ui/src/app/domain/global-state.model.ts +++ b/vripper-web-ui/src/app/domain/global-state.model.ts @@ -1,4 +1,4 @@ export class GlobalState { - constructor(public running: number, public remaining: number, public error: number) { + constructor(public running: number, public remaining: number, public error: number, public loggedUser: string, public downloadSpeed: string) { } } diff --git a/vripper-ui/src/app/domain/multi-post.model.ts b/vripper-web-ui/src/app/domain/multi-post.model.ts similarity index 100% rename from vripper-ui/src/app/domain/multi-post.model.ts rename to vripper-web-ui/src/app/domain/multi-post.model.ts diff --git a/vripper-ui/src/app/domain/photo-model.ts b/vripper-web-ui/src/app/domain/photo-model.ts similarity index 100% rename from vripper-ui/src/app/domain/photo-model.ts rename to vripper-web-ui/src/app/domain/photo-model.ts diff --git a/vripper-ui/src/app/domain/post-id.model.ts b/vripper-web-ui/src/app/domain/post-id.model.ts similarity index 100% rename from vripper-ui/src/app/domain/post-id.model.ts rename to vripper-web-ui/src/app/domain/post-id.model.ts diff --git a/vripper-ui/src/app/domain/post-state.model.ts b/vripper-web-ui/src/app/domain/post-state.model.ts similarity index 77% rename from vripper-ui/src/app/domain/post-state.model.ts rename to vripper-web-ui/src/app/domain/post-state.model.ts index 78b5fc9f..7a79b2dd 100644 --- a/vripper-ui/src/app/domain/post-state.model.ts +++ b/vripper-web-ui/src/app/domain/post-state.model.ts @@ -1,16 +1,13 @@ export class Post { constructor( public postId: string, - public title: string, + public postTitle: string, public progress: number, public status: string, - public removed: boolean, public url: string, public done: number, public total: number, public hosts: string[], - public thanked: boolean, - public previews: string[], public metadata: Metadata, public addedOn: string, public rank: number diff --git a/vripper-ui/src/app/domain/remove-all-response.model.ts b/vripper-web-ui/src/app/domain/remove-all-response.model.ts similarity index 100% rename from vripper-ui/src/app/domain/remove-all-response.model.ts rename to vripper-web-ui/src/app/domain/remove-all-response.model.ts diff --git a/vripper-ui/src/app/domain/remove-response.model.ts b/vripper-web-ui/src/app/domain/remove-response.model.ts similarity index 100% rename from vripper-ui/src/app/domain/remove-response.model.ts rename to vripper-web-ui/src/app/domain/remove-response.model.ts diff --git a/vripper-ui/src/app/domain/rename-post.model.ts b/vripper-web-ui/src/app/domain/rename-post.model.ts similarity index 100% rename from vripper-ui/src/app/domain/rename-post.model.ts rename to vripper-web-ui/src/app/domain/rename-post.model.ts diff --git a/vripper-web-ui/src/app/domain/settings.model.ts b/vripper-web-ui/src/app/domain/settings.model.ts new file mode 100644 index 00000000..1012847a --- /dev/null +++ b/vripper-web-ui/src/app/domain/settings.model.ts @@ -0,0 +1,32 @@ +export interface Settings { + desktopClipboard: boolean; + maxEventLog: number; + connectionSettings: ConnectionSettings; + downloadSettings: DownloadSettings; + viperSettings: ViperSettings; +} + +export interface ConnectionSettings { + maxThreads: number; + maxTotalThreads: number; + timeout: number; + maxAttempts: number; +} + +export interface DownloadSettings { + downloadPath: string; + autoStart: boolean; + forceOrder: boolean; + forumSubfolder: boolean; + threadSubLocation: boolean; + clearCompleted: boolean; + appendPostId: boolean; +} + +export interface ViperSettings { + login: boolean; + username: string; + password: string; + thanks: boolean; + proxy: string; +} diff --git a/vripper-ui/src/app/domain/vr-post-parse.model.ts b/vripper-web-ui/src/app/domain/vr-post-parse.model.ts similarity index 100% rename from vripper-ui/src/app/domain/vr-post-parse.model.ts rename to vripper-web-ui/src/app/domain/vr-post-parse.model.ts diff --git a/vripper-ui/src/app/domain/ws-message.model.ts b/vripper-web-ui/src/app/domain/ws-message.model.ts similarity index 100% rename from vripper-ui/src/app/domain/ws-message.model.ts rename to vripper-web-ui/src/app/domain/ws-message.model.ts diff --git a/vripper-web-ui/src/app/event-log/event-log.component.html b/vripper-web-ui/src/app/event-log/event-log.component.html new file mode 100644 index 00000000..ba1b0df4 --- /dev/null +++ b/vripper-web-ui/src/app/event-log/event-log.component.html @@ -0,0 +1,4 @@ +
+ + +
diff --git a/vripper-ui/src/app/event-log/event-log.component.ts b/vripper-web-ui/src/app/event-log/event-log.component.ts similarity index 87% rename from vripper-ui/src/app/event-log/event-log.component.ts rename to vripper-web-ui/src/app/event-log/event-log.component.ts index c4f1dc15..f8018bae 100644 --- a/vripper-ui/src/app/event-log/event-log.component.ts +++ b/vripper-web-ui/src/app/event-log/event-log.component.ts @@ -1,10 +1,11 @@ -import {ChangeDetectionStrategy, Component, NgZone, OnDestroy} from '@angular/core'; +import {ChangeDetectionStrategy, Component, NgZone, OnDestroy, ViewChild} from '@angular/core'; import {GridOptions} from 'ag-grid-community'; import {WsConnectionService} from '../services/ws-connection.service'; import {EventLogDatasource} from './event-log.datasource'; import {StatusRendererNative} from '../grid-custom-cells/status-renderer.native'; import {EventLogService} from '../services/event-log.service'; import {EventMessageRendererNative} from '../grid-custom-cells/event-message-renderer.native'; +import {AgGridAngular} from "ag-grid-angular"; @Component({ selector: 'app-event-log', @@ -13,7 +14,8 @@ import {EventMessageRendererNative} from '../grid-custom-cells/event-message-ren }) export class EventLogComponent implements OnDestroy { gridOptions: GridOptions; - dataSource: EventLogDatasource; + @ViewChild('agGrid') agGrid!: AgGridAngular; + dataSource!: EventLogDatasource; constructor( private wsConnection: WsConnectionService, @@ -77,7 +79,7 @@ export class EventLogComponent implements OnDestroy { headerHeight: 35, animateRows: true, rowSelection: 'single', - rowDeselection: true, + suppressRowDeselection: false, rowData: [], components: { nativeStatusCellRenderer: StatusRendererNative, @@ -85,14 +87,13 @@ export class EventLogComponent implements OnDestroy { }, overlayLoadingTemplate: '', overlayNoRowsTemplate: '', - getRowNodeId: data => data['id'], + getRowId: row => row.data['id'], onGridReady: () => { - this.eventLogService.setGridApi(this.gridOptions.api); - this.dataSource = new EventLogDatasource(this.wsConnection, this.gridOptions, this.zone); + this.eventLogService.setGridApi(this.agGrid.api); + this.dataSource = new EventLogDatasource(this.wsConnection, this.agGrid.api, this.zone); this.dataSource.connect(); }, - onRowDataUpdated: () => this.eventLogService.setCount(this.gridOptions.api.getDisplayedRowCount()), - onRowDataChanged: () => this.eventLogService.setCount(this.gridOptions.api.getDisplayedRowCount()), + onRowDataUpdated: () => this.eventLogService.setCount(this.agGrid.api.getDisplayedRowCount()), }; } diff --git a/vripper-ui/src/app/event-log/event-log.datasource.ts b/vripper-web-ui/src/app/event-log/event-log.datasource.ts similarity index 62% rename from vripper-ui/src/app/event-log/event-log.datasource.ts rename to vripper-web-ui/src/app/event-log/event-log.datasource.ts index 1978d077..bbc253e0 100644 --- a/vripper-ui/src/app/event-log/event-log.datasource.ts +++ b/vripper-web-ui/src/app/event-log/event-log.datasource.ts @@ -1,42 +1,42 @@ import {Subscription} from 'rxjs'; import {WsConnectionService} from '../services/ws-connection.service'; -import {GridOptions, RowNode} from 'ag-grid-community'; +import {GridApi, IRowNode} from 'ag-grid-community'; import {NgZone} from '@angular/core'; import {EventLog} from '../domain/event.model'; export class EventLogDatasource { subscriptions: Subscription[] = []; - constructor(private ws: WsConnectionService, private gridOptions: GridOptions, private zone: NgZone) { + constructor(private ws: WsConnectionService, private gridApi: GridApi, private zone: NgZone) { } connect() { this.subscriptions.push(this.ws.events$.subscribe((e: EventLog[]) => { this.zone.run(() => { - const toAdd = []; - const toUpdate = []; + const toAdd: EventLog[] = []; + const toUpdate: EventLog[] = []; e.forEach(v => { - if (this.gridOptions.api.getRowNode(v.id.toString(10)) == null) { + if (this.gridApi.getRowNode(v.id.toString(10)) == null) { toAdd.push(v); } else { toUpdate.push(v); } }); - this.gridOptions.api.applyTransaction({update: toUpdate, add: toAdd}); + this.gridApi.applyTransaction({update: toUpdate, add: toAdd}); }); })); this.subscriptions.push(this.ws.eventsRemove$.subscribe((e: number[]) => { this.zone.run(() => { - const toRemove = []; + const toRemove: any[] = []; e.forEach(v => { - const rowNode: RowNode = this.gridOptions.api.getRowNode(v.toString(10)); + const rowNode: IRowNode | undefined = this.gridApi.getRowNode(v.toString(10)); if (rowNode != null) { toRemove.push(rowNode.data); } return; }); - this.gridOptions.api.applyTransaction({remove: toRemove}); + this.gridApi.applyTransaction({remove: toRemove}); }); })); } diff --git a/vripper-ui/src/app/event-log/message-dialog/event-log-message-dialog.component.html b/vripper-web-ui/src/app/event-log/message-dialog/event-log-message-dialog.component.html similarity index 100% rename from vripper-ui/src/app/event-log/message-dialog/event-log-message-dialog.component.html rename to vripper-web-ui/src/app/event-log/message-dialog/event-log-message-dialog.component.html diff --git a/vripper-ui/src/app/event-log/message-dialog/event-log-message-dialog.component.ts b/vripper-web-ui/src/app/event-log/message-dialog/event-log-message-dialog.component.ts similarity index 100% rename from vripper-ui/src/app/event-log/message-dialog/event-log-message-dialog.component.ts rename to vripper-web-ui/src/app/event-log/message-dialog/event-log-message-dialog.component.ts diff --git a/vripper-ui/src/app/grid-custom-cells/collector-actions-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/collector-actions-renderer.native.ts similarity index 89% rename from vripper-ui/src/app/grid-custom-cells/collector-actions-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/collector-actions-renderer.native.ts index 4f52049d..615b9370 100644 --- a/vripper-ui/src/app/grid-custom-cells/collector-actions-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/collector-actions-renderer.native.ts @@ -3,14 +3,14 @@ import {MultiPostService} from '../services/multi-post.service'; import {MultiPostModel} from '../domain/multi-post.model'; export class CollectorActionsRendererNative implements ICellRendererComp { - private gui: HTMLElement; - private selectSpan: HTMLSpanElement; - private selectButton: HTMLButtonElement; - private deleteSpan: HTMLSpanElement; - private deleteButton: HTMLButtonElement; - private multiPostService: MultiPostService; - private multiPostModel: MultiPostModel; - private gridApi: GridApi; + private gui!: HTMLElement; + private selectSpan!: HTMLSpanElement; + private selectButton!: HTMLButtonElement; + private deleteSpan!: HTMLSpanElement; + private deleteButton!: HTMLButtonElement; + private multiPostService!: MultiPostService; + private multiPostModel!: MultiPostModel; + private gridApi!: GridApi; destroy(): void { this.selectButton.removeEventListener('click', () => this.multiPostService.selectItems(this.multiPostModel)); diff --git a/vripper-ui/src/app/grid-custom-cells/event-message-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/event-message-renderer.native.ts similarity index 88% rename from vripper-ui/src/app/grid-custom-cells/event-message-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/event-message-renderer.native.ts index ca5c755e..e444d033 100644 --- a/vripper-ui/src/app/grid-custom-cells/event-message-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/event-message-renderer.native.ts @@ -3,13 +3,13 @@ import {EventLog} from '../domain/event.model'; import {EventLogService} from '../services/event-log.service'; export class EventMessageRendererNative implements ICellRendererComp { - private gui: HTMLElement; - private viewSpan: HTMLSpanElement; - private viewButton: HTMLButtonElement; - private eventLogService: EventLogService; - private eventLog: EventLog; - private gridApi: GridApi; - private text: HTMLSpanElement; + private gui!: HTMLElement; + private viewSpan!: HTMLSpanElement; + private viewButton!: HTMLButtonElement; + private eventLogService!: EventLogService; + private eventLog!: EventLog; + private gridApi!: GridApi; + private text!: HTMLSpanElement; destroy(): void { this.viewButton.removeEventListener('click', () => this.eventLogService.openDialog(this.eventLog.message)); diff --git a/vripper-ui/src/app/grid-custom-cells/post-added-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/post-added-renderer.native.ts similarity index 81% rename from vripper-ui/src/app/grid-custom-cells/post-added-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/post-added-renderer.native.ts index 768cb6ce..dbf30b68 100644 --- a/vripper-ui/src/app/grid-custom-cells/post-added-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/post-added-renderer.native.ts @@ -1,13 +1,13 @@ -import {GridApi, ICellRendererComp, ICellRendererParams, RowNode} from 'ag-grid-community'; +import {GridApi, ICellRendererComp, ICellRendererParams, IRowNode} from 'ag-grid-community'; import {PostContextMenuService} from '../services/post-context-menu.service'; export class PostAddedRendererNative implements ICellRendererComp { - private gui: HTMLElement; - private addedOn: string; - private text: HTMLSpanElement; - private gridApi: GridApi; - private node: RowNode; - private contextMenuService: PostContextMenuService; + private gui!: HTMLElement; + private addedOn!: string; + private text!: HTMLSpanElement; + private gridApi!: GridApi; + private node!: IRowNode; + private contextMenuService!: PostContextMenuService; destroy(): void { if (this.gui) { diff --git a/vripper-ui/src/app/grid-custom-cells/post-alt-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/post-alt-renderer.native.ts similarity index 83% rename from vripper-ui/src/app/grid-custom-cells/post-alt-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/post-alt-renderer.native.ts index 74de5b5b..72a24a2e 100644 --- a/vripper-ui/src/app/grid-custom-cells/post-alt-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/post-alt-renderer.native.ts @@ -1,13 +1,13 @@ -import {GridApi, ICellRendererComp, ICellRendererParams, RowNode} from 'ag-grid-community'; +import {GridApi, ICellRendererComp, ICellRendererParams, IRowNode} from 'ag-grid-community'; import {PostContextMenuService} from '../services/post-context-menu.service'; export class PostAltRendererNative implements ICellRendererComp { - private gui: HTMLElement; - private names: string[]; - private text: HTMLSpanElement; - private gridApi: GridApi; - private node: RowNode; - private contextMenuService: PostContextMenuService; + private gui!: HTMLElement; + private names!: string[]; + private text!: HTMLSpanElement; + private gridApi!: GridApi; + private node!: IRowNode; + private contextMenuService!: PostContextMenuService; destroy(): void { if (this.gui) { diff --git a/vripper-ui/src/app/grid-custom-cells/post-files-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/post-files-renderer.native.ts similarity index 83% rename from vripper-ui/src/app/grid-custom-cells/post-files-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/post-files-renderer.native.ts index 62694b38..742f5359 100644 --- a/vripper-ui/src/app/grid-custom-cells/post-files-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/post-files-renderer.native.ts @@ -1,16 +1,16 @@ -import {GridApi, ICellRendererComp, ICellRendererParams, RowNode} from 'ag-grid-community'; +import {GridApi, ICellRendererComp, ICellRendererParams, IRowNode} from 'ag-grid-community'; import {PostContextMenuService} from '../services/post-context-menu.service'; export class PostFilesRendererNative implements ICellRendererComp { - private gui: HTMLElement; - private done: number; - private total: number; - private hosts: string[]; - private text: HTMLElement; - private gridApi: GridApi; - private node: RowNode; - private contextMenuService: PostContextMenuService; + private gui!: HTMLElement; + private done!: number; + private total!: number; + private hosts!: string[]; + private text!: HTMLElement; + private gridApi!: GridApi; + private node!: IRowNode; + private contextMenuService!: PostContextMenuService; destroy(): void { if (this.gui) { diff --git a/vripper-ui/src/app/grid-custom-cells/post-order-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/post-order-renderer.native.ts similarity index 82% rename from vripper-ui/src/app/grid-custom-cells/post-order-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/post-order-renderer.native.ts index 7be5e5fc..57b540e4 100644 --- a/vripper-ui/src/app/grid-custom-cells/post-order-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/post-order-renderer.native.ts @@ -1,13 +1,13 @@ -import {GridApi, ICellRendererComp, ICellRendererParams, RowNode} from 'ag-grid-community'; +import {GridApi, ICellRendererComp, ICellRendererParams, IRowNode} from 'ag-grid-community'; import {PostContextMenuService} from '../services/post-context-menu.service'; export class PostOrderRendererNative implements ICellRendererComp { - private gui: HTMLElement; - private order: number; - private text: HTMLSpanElement; - private gridApi: GridApi; - private node: RowNode; - private contextMenuService: PostContextMenuService; + private gui!: HTMLElement; + private order!: number; + private text!: HTMLSpanElement; + private gridApi!: GridApi; + private node!: IRowNode; + private contextMenuService!: PostContextMenuService; destroy(): void { if (this.gui) { diff --git a/vripper-ui/src/app/grid-custom-cells/post-progress-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/post-progress-renderer.native.ts similarity index 75% rename from vripper-ui/src/app/grid-custom-cells/post-progress-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/post-progress-renderer.native.ts index a14dc580..72e3c82e 100644 --- a/vripper-ui/src/app/grid-custom-cells/post-progress-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/post-progress-renderer.native.ts @@ -1,20 +1,20 @@ -import {GridApi, ICellRendererParams, RowNode} from 'ag-grid-community'; +import {GridApi, ICellRendererParams, IRowNode} from 'ag-grid-community'; import {PostContextMenuService} from '../services/post-context-menu.service'; import {ProgressRendererNative} from './progress-renderer.native'; export class PostProgressRendererNative extends ProgressRendererNative { - private gridApi: GridApi; - private node: RowNode; - private contextMenuService: PostContextMenuService; + private gridApi!: GridApi; + private node!: IRowNode; + private contextMenuService!: PostContextMenuService; - destroy(): void { + override destroy(): void { if (this.gui) { this.gui.removeEventListener('contextmenu', this.context.bind(this)); } } - init(params: ICellRendererParams): void { + override init(params: ICellRendererParams): void { super.init(params); // @ts-ignore this.contextMenuService = params.contextMenuService; diff --git a/vripper-ui/src/app/grid-custom-cells/post-status-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/post-status-renderer.native.ts similarity index 74% rename from vripper-ui/src/app/grid-custom-cells/post-status-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/post-status-renderer.native.ts index 816d2dfe..508ba8ca 100644 --- a/vripper-ui/src/app/grid-custom-cells/post-status-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/post-status-renderer.native.ts @@ -1,20 +1,20 @@ import {StatusRendererNative} from './status-renderer.native'; -import {GridApi, ICellRendererParams, RowNode} from 'ag-grid-community'; +import {GridApi, ICellRendererParams, IRowNode} from 'ag-grid-community'; import {PostContextMenuService} from '../services/post-context-menu.service'; export class PostStatusRendererNative extends StatusRendererNative { - private gridApi: GridApi; - private node: RowNode; - private contextMenuService: PostContextMenuService; + private gridApi!: GridApi; + private node!: IRowNode; + private contextMenuService!: PostContextMenuService; - destroy(): void { + override destroy(): void { if (this.gui) { this.gui.removeEventListener('contextmenu', this.context.bind(this)); } } - init(params: ICellRendererParams): void { + override init(params: ICellRendererParams): void { super.init(params); // @ts-ignore this.contextMenuService = params.contextMenuService; diff --git a/vripper-ui/src/app/grid-custom-cells/progress-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/progress-renderer.native.ts similarity index 87% rename from vripper-ui/src/app/grid-custom-cells/progress-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/progress-renderer.native.ts index 0be9cb35..e6a6688e 100644 --- a/vripper-ui/src/app/grid-custom-cells/progress-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/progress-renderer.native.ts @@ -2,10 +2,10 @@ import {ICellRendererComp, ICellRendererParams} from 'ag-grid-community'; export class ProgressRendererNative implements ICellRendererComp { - protected gui: HTMLElement; - protected progressBackDiv: HTMLElement; - protected progressFrontDiv: HTMLElement; - private progress: number; + protected gui!: HTMLElement; + protected progressBackDiv!: HTMLElement; + protected progressFrontDiv!: HTMLElement; + private progress!: number; destroy(): void { } diff --git a/vripper-ui/src/app/grid-custom-cells/status-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/status-renderer.native.ts similarity index 91% rename from vripper-ui/src/app/grid-custom-cells/status-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/status-renderer.native.ts index 6a929c49..a6ae032f 100644 --- a/vripper-ui/src/app/grid-custom-cells/status-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/status-renderer.native.ts @@ -2,12 +2,12 @@ import {ICellRendererComp, ICellRendererParams} from 'ag-grid-community'; export class StatusRendererNative implements ICellRendererComp { - protected gui: HTMLElement; - private container: HTMLElement; - private icon: HTMLElement; - private text: HTMLElement; - private status: string; - private currentStatusClass; + protected gui!: HTMLElement; + private container!: HTMLElement; + private icon!: HTMLElement; + private text!: HTMLElement; + private status!: string; + private currentStatusClass: string | undefined; destroy(): void { } @@ -26,8 +26,8 @@ export class StatusRendererNative implements ICellRendererComp { this.icon = document.createElement('span'); this.currentStatusClass = this.getClass(this.status); - this.icon.classList.add('native-status-icon', this.currentStatusClass); - this.icon.innerHTML = this.getIcon(this.status); + this.icon.classList.add('native-status-icon', this.currentStatusClass ?? ''); + this.icon.innerHTML = this.getIcon(this.status) ?? ''; this.text = document.createElement('span'); this.text.classList.add('native-status-text'); @@ -41,9 +41,8 @@ export class StatusRendererNative implements ICellRendererComp { const oldStatusClass = this.currentStatusClass; this.status = params.value; this.currentStatusClass = this.getClass(this.status); - this.icon.classList.replace(oldStatusClass, this.currentStatusClass); + this.icon.classList.replace(oldStatusClass ?? '', this.currentStatusClass ?? ''); this.icon.innerHTML = this.getIcon(this.status); - this.text.innerText = this.titleCase(this.status); return true; } @@ -83,6 +82,8 @@ export class StatusRendererNative implements ICellRendererComp { return ` `; + default: + return ''; } } @@ -93,7 +94,7 @@ export class StatusRendererNative implements ICellRendererComp { case 'DOWNLOADING': case 'PROCESSING': return 'downloading'; - case 'COMPLETE': + case 'FINISHED': case 'DONE': return 'complete'; case 'ERROR': @@ -102,6 +103,8 @@ export class StatusRendererNative implements ICellRendererComp { return 'stopped'; case 'PARTIAL': return 'downloading'; + default: + return '' } } } diff --git a/vripper-ui/src/app/grid-custom-cells/title-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/title-renderer.native.ts similarity index 88% rename from vripper-ui/src/app/grid-custom-cells/title-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/title-renderer.native.ts index 95b380aa..6b04dbe4 100644 --- a/vripper-ui/src/app/grid-custom-cells/title-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/title-renderer.native.ts @@ -1,4 +1,4 @@ -import {GridApi, ICellRendererComp, ICellRendererParams, RowNode} from 'ag-grid-community'; +import {GridApi, ICellRendererComp, ICellRendererParams, IRowNode} from 'ag-grid-community'; import {PostContextMenuService} from '../services/post-context-menu.service'; import {Overlay, OverlayPositionBuilder, OverlayRef} from '@angular/cdk/overlay'; import {ComponentPortal} from '@angular/cdk/portal'; @@ -7,17 +7,17 @@ import {ComponentRef, NgZone} from '@angular/core'; export class TitleRendererNative implements ICellRendererComp { tooltipPortal = new ComponentPortal(AppPreviewComponent); - appPreview: string[]; - private gui: HTMLElement; - private text: HTMLSpanElement; - private gridApi: GridApi; - private node: RowNode; - private contextMenuService: PostContextMenuService; - private icon: HTMLSpanElement; - private overlayPositionBuilder: OverlayPositionBuilder; - private overlay: Overlay; - private zone: NgZone; - private overlayRef: OverlayRef; + appPreview!: string[]; + private gui!: HTMLElement; + private text!: HTMLSpanElement; + private gridApi!: GridApi; + private node!: IRowNode; + private contextMenuService!: PostContextMenuService; + private icon!: HTMLSpanElement; + private overlayPositionBuilder!: OverlayPositionBuilder; + private overlay!: Overlay; + private zone!: NgZone; + private overlayRef!: OverlayRef; destroy(): void { if (this.gui) { diff --git a/vripper-ui/src/app/grid-custom-cells/url-renderer.native.ts b/vripper-web-ui/src/app/grid-custom-cells/url-renderer.native.ts similarity index 67% rename from vripper-ui/src/app/grid-custom-cells/url-renderer.native.ts rename to vripper-web-ui/src/app/grid-custom-cells/url-renderer.native.ts index ac4c6f23..bf19e6f3 100644 --- a/vripper-ui/src/app/grid-custom-cells/url-renderer.native.ts +++ b/vripper-web-ui/src/app/grid-custom-cells/url-renderer.native.ts @@ -1,13 +1,11 @@ import {ICellRendererComp, ICellRendererParams} from 'ag-grid-community'; -import {ElectronService} from 'ngx-electron'; export class UrlRendererNative implements ICellRendererComp { - private electronService: ElectronService; - private gui: HTMLElement; - private text: HTMLSpanElement; - private link: HTMLAnchorElement; - private url: string; + private gui!: HTMLElement; + private text!: HTMLSpanElement; + private link!: HTMLAnchorElement; + private url!: string; destroy(): void { if (this.link) { @@ -20,9 +18,6 @@ export class UrlRendererNative implements ICellRendererComp { } init(params: ICellRendererParams): void { - // @ts-ignore - this.electronService = params.electronService; - this.url = params.value; this.gui = document.createElement('div'); this.gui.classList.add('text-cell'); @@ -43,10 +38,6 @@ export class UrlRendererNative implements ICellRendererComp { } goTo() { - if (this.electronService.isElectronApp) { - this.electronService.shell.openExternal(this.url).then(); - } else { - window.open(this.url, '_blank'); - } + window.open(this.url, '_blank'); } } diff --git a/vripper-ui/src/app/home/home.component.html b/vripper-web-ui/src/app/home/home.component.html similarity index 100% rename from vripper-ui/src/app/home/home.component.html rename to vripper-web-ui/src/app/home/home.component.html diff --git a/vripper-ui/src/app/home/home.component.scss b/vripper-web-ui/src/app/home/home.component.scss similarity index 89% rename from vripper-ui/src/app/home/home.component.scss rename to vripper-web-ui/src/app/home/home.component.scss index c9244f6a..be806752 100644 --- a/vripper-ui/src/app/home/home.component.scss +++ b/vripper-web-ui/src/app/home/home.component.scss @@ -2,10 +2,6 @@ table { width: 100%; } -:host ::ng-deep .mat-tab-body-wrapper { - height: 100%; -} - .app-custom-plain-container-with-desc { align-items: center; display: flex; @@ -44,7 +40,7 @@ table { height: auto; margin-right: 2px; width: 200px; - align-self: start; + align-self: flex-start; } } diff --git a/vripper-ui/src/app/home/home.component.ts b/vripper-web-ui/src/app/home/home.component.ts similarity index 57% rename from vripper-ui/src/app/home/home.component.ts rename to vripper-web-ui/src/app/home/home.component.ts index b5f78d87..168d7072 100644 --- a/vripper-ui/src/app/home/home.component.ts +++ b/vripper-web-ui/src/app/home/home.component.ts @@ -1,7 +1,6 @@ import {Subscription} from 'rxjs'; import {LinkCollectorService} from '../services/link-collector.service'; import {ServerService} from '../services/server-service'; -import {ClipboardService} from '../services/clipboard.service'; import {ChangeDetectionStrategy, Component, NgZone, OnDestroy, OnInit} from '@angular/core'; import {MatDialog} from '@angular/material/dialog'; import {MatSnackBar} from '@angular/material/snack-bar'; @@ -17,11 +16,9 @@ import {HomeTabsService} from '../services/home-tabs.service'; styleUrls: ['./home.component.scss'], changeDetection: ChangeDetectionStrategy.OnPush }) -export class HomeComponent implements OnInit, OnDestroy { - clipboardSub: Subscription; +export class HomeComponent { constructor( - private clipboardService: ClipboardService, public dialog: MatDialog, private httpClient: HttpClient, private serverService: ServerService, @@ -34,33 +31,6 @@ export class HomeComponent implements OnInit, OnDestroy { ) { } - ngOnInit() { - this.clipboardSub = this.clipboardService.links.subscribe(e => { - this.ngZone.run(() => { - this.httpClient - .post<{ threadId: string; postId: string }>(this.serverService.baseUrl + '/post', {url: e}) - .subscribe( - response => { - this._snackBar.open('Clipboard scan complete', null, { - duration: 5000 - }); - }, - error => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { - duration: 5000 - }); - } - ); - }); - }); - } - - ngOnDestroy() { - if (this.clipboardSub != null) { - this.clipboardSub.unsubscribe(); - } - } - onTabChange($event: MatTabChangeEvent) { this.homeTabsService.setIndex($event.index); } diff --git a/vripper-ui/src/app/material.module.ts b/vripper-web-ui/src/app/material.module.ts similarity index 100% rename from vripper-ui/src/app/material.module.ts rename to vripper-web-ui/src/app/material.module.ts diff --git a/vripper-ui/src/app/multi-post-grid/multi-post-grid-data.source.ts b/vripper-web-ui/src/app/multi-post-grid/multi-post-grid-data.source.ts similarity index 63% rename from vripper-ui/src/app/multi-post-grid/multi-post-grid-data.source.ts rename to vripper-web-ui/src/app/multi-post-grid/multi-post-grid-data.source.ts index 3c47cecb..78971114 100644 --- a/vripper-ui/src/app/multi-post-grid/multi-post-grid-data.source.ts +++ b/vripper-web-ui/src/app/multi-post-grid/multi-post-grid-data.source.ts @@ -1,7 +1,7 @@ import {MultiPostModel} from '../domain/multi-post.model'; import {Subscription} from 'rxjs'; import {WsConnectionService} from '../services/ws-connection.service'; -import {GridOptions, RowNode} from 'ag-grid-community'; +import {GridApi, IRowNode, RowNode} from 'ag-grid-community'; import {NgZone} from '@angular/core'; export class MultiPostGridDataSource { @@ -9,7 +9,7 @@ export class MultiPostGridDataSource { constructor( private ws: WsConnectionService, - private gridOptions: GridOptions, + private gridApi: GridApi, private zone: NgZone ) { } @@ -17,31 +17,31 @@ export class MultiPostGridDataSource { connect() { this.subscriptions.push(this.ws.multiPosts$.subscribe((e: MultiPostModel[]) => { this.zone.run(() => { - const toAdd = []; - const toUpdate = []; + const toAdd: MultiPostModel[] = []; + const toUpdate: MultiPostModel[] = []; e.forEach(v => { - const rowNode: RowNode = this.gridOptions.api.getRowNode(v.threadId); + const rowNode: IRowNode | undefined = this.gridApi.getRowNode(v.threadId); if (rowNode == null) { toAdd.push(v); } else { toUpdate.push(v); } }); - this.gridOptions.api.applyTransaction({update: toUpdate, add: toAdd}); + this.gridApi.applyTransaction({update: toUpdate, add: toAdd}); }); })); this.subscriptions.push(this.ws.queuedRemove$.subscribe((e: string[]) => { this.zone.run(() => { - const toRemove = []; + const toRemove: MultiPostModel[] = []; e.forEach(v => { - const rowNode: RowNode = this.gridOptions.api.getRowNode(v); - if (rowNode != null) { + const rowNode: IRowNode | undefined = this.gridApi.getRowNode(v); + if (rowNode != null && rowNode.data != null) { toRemove.push(rowNode.data); } return; }); - this.gridOptions.api.applyTransaction({remove: toRemove}); + this.gridApi.applyTransaction({remove: toRemove}); }); })); } diff --git a/vripper-web-ui/src/app/multi-post-grid/multi-post-grid.component.html b/vripper-web-ui/src/app/multi-post-grid/multi-post-grid.component.html new file mode 100644 index 00000000..ba1b0df4 --- /dev/null +++ b/vripper-web-ui/src/app/multi-post-grid/multi-post-grid.component.html @@ -0,0 +1,4 @@ +
+ + +
diff --git a/vripper-ui/src/app/multi-post-grid/multi-post-grid.component.ts b/vripper-web-ui/src/app/multi-post-grid/multi-post-grid.component.ts similarity index 75% rename from vripper-ui/src/app/multi-post-grid/multi-post-grid.component.ts rename to vripper-web-ui/src/app/multi-post-grid/multi-post-grid.component.ts index f7c1fee2..2f27a467 100644 --- a/vripper-ui/src/app/multi-post-grid/multi-post-grid.component.ts +++ b/vripper-web-ui/src/app/multi-post-grid/multi-post-grid.component.ts @@ -1,12 +1,12 @@ import {LinkCollectorService} from '../services/link-collector.service'; import {MultiPostGridDataSource} from './multi-post-grid-data.source'; -import {ChangeDetectionStrategy, Component, NgZone, OnDestroy} from '@angular/core'; +import {ChangeDetectionStrategy, Component, NgZone, OnDestroy, ViewChild} from '@angular/core'; import {GridOptions} from 'ag-grid-community'; import {WsConnectionService} from '../services/ws-connection.service'; import {CollectorActionsRendererNative} from '../grid-custom-cells/collector-actions-renderer.native'; import {MultiPostService} from '../services/multi-post.service'; import {UrlRendererNative} from '../grid-custom-cells/url-renderer.native'; -import {ElectronService} from 'ngx-electron'; +import {AgGridAngular} from "ag-grid-angular"; @Component({ selector: 'app-grab-queue', @@ -16,14 +16,15 @@ import {ElectronService} from 'ngx-electron'; export class MultiPostGridComponent implements OnDestroy { gridOptions: GridOptions; - dataSource: MultiPostGridDataSource; + + @ViewChild('agGrid') agGrid!: AgGridAngular; + dataSource!: MultiPostGridDataSource; constructor( private wsConnection: WsConnectionService, private zone: NgZone, private linkCollectorService: LinkCollectorService, - private grabQueueService: MultiPostService, - private electronService: ElectronService) { + private grabQueueService: MultiPostService) { this.gridOptions = { columnDefs: [ { @@ -31,9 +32,6 @@ export class MultiPostGridComponent implements OnDestroy { field: 'link', sort: 'asc', cellRenderer: 'urlCellRenderer', - cellRendererParams: { - electronService: this.electronService - }, flex: 1 }, { headerName: 'Count', @@ -65,14 +63,13 @@ export class MultiPostGridComponent implements OnDestroy { }, overlayLoadingTemplate: '', overlayNoRowsTemplate: '', - getRowNodeId: data => data['threadId'], - onGridReady: () => { - this.linkCollectorService.setGridApi(this.gridOptions.api); - this.dataSource = new MultiPostGridDataSource(this.wsConnection, this.gridOptions, this.zone); + getRowId: row => row.data['threadId'], + onGridReady: (prams) => { + this.linkCollectorService.setGridApi(this.agGrid.api); + this.dataSource = new MultiPostGridDataSource(this.wsConnection, this.agGrid.api, this.zone); this.dataSource.connect(); }, - onRowDataUpdated: () => this.linkCollectorService.setCount(this.gridOptions.api.getDisplayedRowCount()), - onRowDataChanged: () => this.linkCollectorService.setCount(this.gridOptions.api.getDisplayedRowCount()), + onRowDataUpdated: () => this.linkCollectorService.setCount(this.agGrid.api.getDisplayedRowCount()), }; } diff --git a/vripper-ui/src/app/multi-post-items/multi-post-items.component.html b/vripper-web-ui/src/app/multi-post-items/multi-post-items.component.html similarity index 100% rename from vripper-ui/src/app/multi-post-items/multi-post-items.component.html rename to vripper-web-ui/src/app/multi-post-items/multi-post-items.component.html diff --git a/vripper-ui/src/app/multi-post-items/multi-post-items.component.scss b/vripper-web-ui/src/app/multi-post-items/multi-post-items.component.scss similarity index 100% rename from vripper-ui/src/app/multi-post-items/multi-post-items.component.scss rename to vripper-web-ui/src/app/multi-post-items/multi-post-items.component.scss diff --git a/vripper-ui/src/app/multi-post-items/multi-post-items.component.ts b/vripper-web-ui/src/app/multi-post-items/multi-post-items.component.ts similarity index 79% rename from vripper-ui/src/app/multi-post-items/multi-post-items.component.ts rename to vripper-web-ui/src/app/multi-post-items/multi-post-items.component.ts index 47d0efcd..d2eb1e17 100644 --- a/vripper-ui/src/app/multi-post-items/multi-post-items.component.ts +++ b/vripper-web-ui/src/app/multi-post-items/multi-post-items.component.ts @@ -3,7 +3,7 @@ import {MultiPostItem} from '../domain/vr-post-parse.model'; import {ChangeDetectionStrategy, Component, Inject, NgZone} from '@angular/core'; import {MAT_DIALOG_DATA, MatDialogRef} from '@angular/material/dialog'; import {MatSnackBar} from '@angular/material/snack-bar'; -import {GridOptions, RowNode} from 'ag-grid-community'; +import {GridApi, GridOptions, IRowNode} from 'ag-grid-community'; import {BehaviorSubject, Subject} from 'rxjs'; import {HttpClient} from '@angular/common/http'; import {ServerService} from '../services/server-service'; @@ -13,7 +13,6 @@ import {PostContextMenuService} from '../services/post-context-menu.service'; import {Overlay, OverlayPositionBuilder} from '@angular/cdk/overlay'; import {UrlRendererNative} from '../grid-custom-cells/url-renderer.native'; import {ValueFormatterParams} from 'ag-grid-community/dist/lib/entities/colDef'; -import {ElectronService} from 'ngx-electron'; @Component({ selector: 'app-multi-post-items', @@ -23,6 +22,7 @@ import {ElectronService} from 'ngx-electron'; }) export class MultiPostItemsComponent { gridOptions: GridOptions; + gridApi!: GridApi; loading: Subject = new BehaviorSubject(true); selectedRowsCount: Subject = new BehaviorSubject(0); @@ -36,8 +36,7 @@ export class MultiPostItemsComponent { private contextMenuService: PostContextMenuService, private overlayPositionBuilder: OverlayPositionBuilder, private overlay: Overlay, - private zone: NgZone, - private electronService: ElectronService + private zone: NgZone ) { this.gridOptions = { columnDefs: [ @@ -67,9 +66,6 @@ export class MultiPostItemsComponent { headerName: 'URL', field: 'url', cellRenderer: 'urlCellRenderer', - cellRendererParams: { - electronService: this.electronService - }, flex: 1 }, { @@ -95,22 +91,25 @@ export class MultiPostItemsComponent { rowData: [], overlayLoadingTemplate: '', overlayNoRowsTemplate: '', - getRowNodeId: data => data['url'], - onGridReady: () => this.fetchMultiPostItems(), - onSelectionChanged: () => this.onSelectionChange(this.gridOptions.api.getSelectedNodes()) + getRowId: row => row.data['url'], + onGridReady: (params) => { + this.gridApi = params.api; + this.fetchMultiPostItems(); + }, + onSelectionChanged: () => this.onSelectionChange(this.gridApi.getSelectedNodes()) }; } - onSelectionChange(data: RowNode[]) { + onSelectionChange(data: IRowNode[]) { this.selectedRowsCount.next(data.length); } - search(event) { - this.gridOptions.api.setQuickFilter(event); + search(event: string) { + this.gridApi.setQuickFilter(event); } submit() { - const data = (this.gridOptions.api.getSelectedRows()).map(e => ({ + const data = (this.gridApi.getSelectedRows()).map(e => ({ postId: e.postId, threadId: this.dialogData.threadId })); @@ -124,12 +123,12 @@ export class MultiPostItemsComponent { }) ) .subscribe( - () => { - }, - error => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { - duration: 5000 - }); + { + error: (error) => { + this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', undefined, { + duration: 5000 + }); + } } ); } @@ -139,13 +138,14 @@ export class MultiPostItemsComponent { .get(this.serverService.baseUrl + '/grab/' + this.dialogData.threadId) .pipe(finalize(() => this.loading.next(false))) .subscribe( - data => { - this.gridOptions.api.applyTransaction({add: data}); - }, - error => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { - duration: 5000 - }); + { + next: (data) => { + this.gridApi.applyTransaction({add: data}); + }, error: (error) => { + this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', undefined, { + duration: 5000 + }); + } } ); } diff --git a/vripper-ui/src/app/photos/photos.component.html b/vripper-web-ui/src/app/photos/photos.component.html similarity index 86% rename from vripper-ui/src/app/photos/photos.component.html rename to vripper-web-ui/src/app/photos/photos.component.html index 6a7c7517..5034ce18 100644 --- a/vripper-ui/src/app/photos/photos.component.html +++ b/vripper-web-ui/src/app/photos/photos.component.html @@ -1,6 +1,6 @@
-

{{ dialogData.title }}

+

{{ dialogData.postTitle }}

diff --git a/vripper-ui/src/app/photos/photos.component.scss b/vripper-web-ui/src/app/photos/photos.component.scss similarity index 100% rename from vripper-ui/src/app/photos/photos.component.scss rename to vripper-web-ui/src/app/photos/photos.component.scss diff --git a/vripper-ui/src/app/photos/photos.component.ts b/vripper-web-ui/src/app/photos/photos.component.ts similarity index 85% rename from vripper-ui/src/app/photos/photos.component.ts rename to vripper-web-ui/src/app/photos/photos.component.ts index 61d2e2f5..f97a8aa6 100644 --- a/vripper-ui/src/app/photos/photos.component.ts +++ b/vripper-web-ui/src/app/photos/photos.component.ts @@ -3,11 +3,10 @@ import {ChangeDetectionStrategy, Component, Inject, NgZone, OnDestroy, OnInit} f import {MAT_DIALOG_DATA, MatDialogRef} from '@angular/material/dialog'; import {PhotosDatasource} from './photos.datasource'; import {Post} from '../domain/post-state.model'; -import {GridOptions} from 'ag-grid-community'; +import {GridApi, GridOptions} from 'ag-grid-community'; import {ProgressRendererNative} from '../grid-custom-cells/progress-renderer.native'; import {StatusRendererNative} from '../grid-custom-cells/status-renderer.native'; import {UrlRendererNative} from '../grid-custom-cells/url-renderer.native'; -import {ElectronService} from 'ngx-electron'; @Component({ selector: 'app-post-detail', @@ -18,14 +17,14 @@ import {ElectronService} from 'ngx-electron'; export class PhotosComponent implements OnInit, OnDestroy { gridOptions: GridOptions; - dataSource: PhotosDatasource; + dataSource!: PhotosDatasource; + private gridApi!: GridApi; constructor( public dialogRef: MatDialogRef, @Inject(MAT_DIALOG_DATA) public dialogData: Post, private wsConnection: WsConnectionService, - private zone: NgZone, - private electronService: ElectronService + private zone: NgZone ) { this.gridOptions = { @@ -40,9 +39,6 @@ export class PhotosComponent implements OnInit, OnDestroy { headerName: 'URL', field: 'url', cellRenderer: 'urlCellRenderer', - cellRendererParams: { - electronService: this.electronService - }, flex: 1 }, { headerName: 'Progress', @@ -73,13 +69,13 @@ export class PhotosComponent implements OnInit, OnDestroy { }, overlayLoadingTemplate: '', overlayNoRowsTemplate: '', - getRowNodeId: (data) => data['url'], - onGridReady: () => { - this.dataSource = new PhotosDatasource(this.wsConnection, this.gridOptions, this.dialogData.postId, this.zone); + getRowId: (row) => row.data['url'], + onGridReady: (params) => { + this.gridApi = params.api; + this.dataSource = new PhotosDatasource(this.wsConnection, this.gridApi, this.dialogData.postId, this.zone); this.dataSource.connect(); }, }; - } ngOnInit() { @@ -92,5 +88,4 @@ export class PhotosComponent implements OnInit, OnDestroy { ngOnDestroy(): void { this.dataSource.disconnect(); } - } diff --git a/vripper-ui/src/app/photos/photos.datasource.ts b/vripper-web-ui/src/app/photos/photos.datasource.ts similarity index 68% rename from vripper-ui/src/app/photos/photos.datasource.ts rename to vripper-web-ui/src/app/photos/photos.datasource.ts index 2032ccaf..886f4d51 100644 --- a/vripper-ui/src/app/photos/photos.datasource.ts +++ b/vripper-web-ui/src/app/photos/photos.datasource.ts @@ -1,15 +1,16 @@ -import {GridOptions} from 'ag-grid-community'; +import {GridApi} from 'ag-grid-community'; import {Subscription} from 'rxjs'; import {WsConnectionService} from '../services/ws-connection.service'; import {NgZone} from '@angular/core'; +import {Photo} from '../domain/photo-model'; export class PhotosDatasource { subscriptions: Subscription[] = []; - postDetailsSub: Subscription; + postDetailsSub!: Subscription; constructor( private ws: WsConnectionService, - private gridOptions: GridOptions, + private gridApi: GridApi, private postId: string, private zone: NgZone ) { @@ -18,16 +19,16 @@ export class PhotosDatasource { connect() { this.postDetailsSub = this.ws.postDetails$(this.postId).subscribe(e => { this.zone.run(() => { - const toAdd = []; - const toUpdate = []; + const toAdd: Photo[] = []; + const toUpdate: Photo[] = []; e.forEach(v => { - if (this.gridOptions.api.getRowNode(v.url) == null) { + if (this.gridApi.getRowNode(v.url) == null) { toAdd.push(v); } else { toUpdate.push(v); } }); - this.gridOptions.api.applyTransaction({update: toUpdate, add: toAdd}); + this.gridApi.applyTransaction({update: toUpdate, add: toAdd}); }); }); } diff --git a/vripper-ui/src/app/posts/context-menu/post-context-menu.component.html b/vripper-web-ui/src/app/posts/context-menu/post-context-menu.component.html similarity index 70% rename from vripper-ui/src/app/posts/context-menu/post-context-menu.component.html rename to vripper-web-ui/src/app/posts/context-menu/post-context-menu.component.html index 61ff852e..210cb20d 100644 --- a/vripper-ui/src/app/posts/context-menu/post-context-menu.component.html +++ b/vripper-web-ui/src/app/posts/context-menu/post-context-menu.component.html @@ -28,21 +28,9 @@ list Photos - - - diff --git a/vripper-ui/src/app/posts/context-menu/post-context-menu.component.scss b/vripper-web-ui/src/app/posts/context-menu/post-context-menu.component.scss similarity index 100% rename from vripper-ui/src/app/posts/context-menu/post-context-menu.component.scss rename to vripper-web-ui/src/app/posts/context-menu/post-context-menu.component.scss diff --git a/vripper-ui/src/app/posts/context-menu/post-context-menu.component.ts b/vripper-web-ui/src/app/posts/context-menu/post-context-menu.component.ts similarity index 52% rename from vripper-ui/src/app/posts/context-menu/post-context-menu.component.ts rename to vripper-web-ui/src/app/posts/context-menu/post-context-menu.component.ts index ba84bb2d..8c94b6b8 100644 --- a/vripper-ui/src/app/posts/context-menu/post-context-menu.component.ts +++ b/vripper-web-ui/src/app/posts/context-menu/post-context-menu.component.ts @@ -1,22 +1,19 @@ import {AppService} from '../../services/app.service'; import {ServerService} from '../../services/server-service'; -import {HttpClient, HttpErrorResponse} from '@angular/common/http'; -import {ElectronService} from 'ngx-electron'; +import {HttpClient} from '@angular/common/http'; import {ChangeDetectionStrategy, Component, NgZone} from '@angular/core'; import {animate, state, style, transition, trigger} from '@angular/animations'; -import {MatDialog, MatDialogConfig} from '@angular/material/dialog'; +import {MatDialog} from '@angular/material/dialog'; import {MatSnackBar} from '@angular/material/snack-bar'; import {PhotosComponent} from '../../photos/photos.component'; import {Observable} from 'rxjs'; import {BreakpointObserver, Breakpoints, BreakpointState} from '@angular/cdk/layout'; import {Post} from '../../domain/post-state.model'; -import {DownloadPath} from '../../domain/download-path.model'; import {ConfirmDialogComponent} from '../../confirmation-component/confirmation-dialog'; import {filter, flatMap} from 'rxjs/operators'; import {RemoveResponse} from '../../domain/remove-response.model'; import {PostContextMenuService} from '../../services/post-context-menu.service'; import {PostsService} from '../../services/posts.service'; -import {AlternativeTitleComponent, AlternativeTitleDialog} from '../alternative-title/alternative-title.component'; import {PostId} from '../../domain/post-id.model'; @Component({ @@ -33,13 +30,11 @@ import {PostId} from '../../domain/post-id.model'; changeDetection: ChangeDetectionStrategy.OnPush }) export class PostContextMenuComponent { - post: Post; + post!: Post; isExtraSmall: Observable = this.breakpointObserver.observe(Breakpoints.XSmall); - fs; - contextMenuService: PostContextMenuService; + contextMenuService!: PostContextMenuService; constructor( - public electronService: ElectronService, private dialog: MatDialog, private breakpointObserver: BreakpointObserver, private httpClient: HttpClient, @@ -49,32 +44,25 @@ export class PostContextMenuComponent { private appService: AppService, private postsDataService: PostsService ) { - if (this.electronService.isElectronApp) { - this.fs = this.electronService.remote.require('fs'); - } } restart() { this.contextMenuService.closePostContextMenu(); this.httpClient.post(this.serverService.baseUrl + '/post/restart', [this.post.postId]).subscribe( - () => { - }, - error => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { - duration: 5000 - }); + { + error: error => { + this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', undefined, { + duration: 5000 + }); + } } ); } goTo() { this.contextMenuService.closePostContextMenu(); - if (this.electronService.isElectronApp) { - this.electronService.shell.openExternal(this.post.url); - } else { - window.open(this.post.url, '_blank'); - } + window.open(this.post.url, '_blank'); } remove() { @@ -92,61 +80,30 @@ export class PostContextMenuComponent { .pipe( filter(e => e === 'yes'), flatMap(() => - this.httpClient.post(this.serverService.baseUrl + '/post/remove', [this.post.postId]) + this.httpClient.post(this.serverService.baseUrl + '/post/remove', [this.post.postId]) ) ) - .subscribe( - data => { - this.postsDataService.remove(data); - }, - error => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { - duration: 5000 - }); + .subscribe({ + next: data => { + this.postsDataService.remove(data); + }, error: error => { + this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', undefined, { + duration: 5000 + }); + } } ); }); } - useAltTitle() { - this.contextMenuService.closePostContextMenu(); - this.httpClient.post(this.serverService.baseUrl + '/post/rename/first', [{ - postId: this.post.postId - }]).subscribe( - () => { - }, - error => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { - duration: 5000 - }); - } - ); - } - - openRenameDialog() { - this.contextMenuService.closePostContextMenu(); - const dialogConfig: MatDialogConfig = { - maxHeight: '100vh', - maxWidth: '100vw', - height: '300px', - width: '60%', - data: {post: this.post} - }; - this.ngZone.run(() => { - this.dialog - .open(AlternativeTitleComponent, dialogConfig); - }); - } - stop() { this.contextMenuService.closePostContextMenu(); - this.httpClient.post(this.serverService.baseUrl + '/post/stop', [this.post.postId]).subscribe( - () => { - }, - error => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { - duration: 5000 - }); + this.httpClient.post(this.serverService.baseUrl + '/post/stop', [this.post.postId]).subscribe({ + error: error => { + this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', undefined, { + duration: 5000 + }); + } } ); } @@ -177,29 +134,4 @@ export class PostContextMenuComponent { }); }); } - - open() { - this.contextMenuService.closePostContextMenu(); - if (!this.electronService.isElectronApp) { - console.error('Cannot open downloader folder, not electron app'); - return; - } - // Request the server to give the correct file location - this.httpClient.get(this.serverService.baseUrl + '/post/path/' + this.post.postId).subscribe( - path => { - if (this.fs.existsSync(path.path)) { - this.electronService.shell.openPath(path.path).then(); - } else { - this._snackBar.open(path.path + ' does not exist, you probably removed it', null, { - duration: 5000 - }); - } - }, - (error: HttpErrorResponse) => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { - duration: 5000 - }); - } - ); - } } diff --git a/vripper-web-ui/src/app/posts/posts.component.html b/vripper-web-ui/src/app/posts/posts.component.html new file mode 100644 index 00000000..ba1b0df4 --- /dev/null +++ b/vripper-web-ui/src/app/posts/posts.component.html @@ -0,0 +1,4 @@ +
+ + +
diff --git a/vripper-ui/src/app/posts/posts.component.ts b/vripper-web-ui/src/app/posts/posts.component.ts similarity index 84% rename from vripper-ui/src/app/posts/posts.component.ts rename to vripper-web-ui/src/app/posts/posts.component.ts index 8d495d74..9a081fad 100644 --- a/vripper-ui/src/app/posts/posts.component.ts +++ b/vripper-web-ui/src/app/posts/posts.component.ts @@ -1,5 +1,5 @@ import {SelectionService} from '../services/selection-service'; -import {ChangeDetectionStrategy, Component, NgZone, OnDestroy} from '@angular/core'; +import {ChangeDetectionStrategy, Component, NgZone, OnDestroy, ViewChild} from '@angular/core'; import {PostsDataSource} from './posts.datasource'; import {WsConnectionService} from '../services/ws-connection.service'; import {GridOptions} from 'ag-grid-community'; @@ -13,6 +13,7 @@ import {Overlay, OverlayPositionBuilder} from '@angular/cdk/overlay'; import {PostsService} from '../services/posts.service'; import {PostAddedRendererNative} from '../grid-custom-cells/post-added-renderer.native'; import {PostOrderRendererNative} from '../grid-custom-cells/post-order-renderer.native'; +import {AgGridAngular} from "ag-grid-angular"; @Component({ selector: 'app-posts', @@ -21,7 +22,8 @@ import {PostOrderRendererNative} from '../grid-custom-cells/post-order-renderer. }) export class PostsComponent implements OnDestroy { gridOptions: GridOptions; - dataSource: PostsDataSource; + dataSource!: PostsDataSource; + @ViewChild('agGrid') agGrid!: AgGridAngular; constructor( private wsConnection: WsConnectionService, @@ -37,7 +39,7 @@ export class PostsComponent implements OnDestroy { columnDefs: [ { headerName: 'Title', - field: 'title', + field: 'postTitle', cellRenderer: 'nativeTitleCellRenderer', cellRendererParams: { contextMenuService: this.contextMenuService, @@ -74,14 +76,6 @@ export class PostsComponent implements OnDestroy { contextMenuService: this.contextMenuService }, flex: 1 - }, { - headerName: 'Alternative titles', - field: 'metadata', - cellRenderer: 'nativeAltCellRenderer', - cellRendererParams: { - contextMenuService: this.contextMenuService - }, - flex: 1 }, { headerName: 'Added On', field: 'addedOn', @@ -110,7 +104,7 @@ export class PostsComponent implements OnDestroy { headerHeight: 35, animateRows: true, rowSelection: 'multiple', - rowDeselection: true, + suppressRowDeselection: false, rowData: [], components: { nativeProgressCellRenderer: PostProgressRendererNative, @@ -123,16 +117,15 @@ export class PostsComponent implements OnDestroy { }, overlayLoadingTemplate: '', overlayNoRowsTemplate: '', - getRowNodeId: data => data['postId'], + getRowId: row => row.data['postId'], onGridReady: () => { - this.dataSource = new PostsDataSource(this.wsConnection, this.gridOptions, this.zone); - this.postsDataService.setGridApi(this.gridOptions.api); + this.dataSource = new PostsDataSource(this.wsConnection, this.agGrid.api, this.zone); + this.postsDataService.setGridApi(this.agGrid.api); this.dataSource.connect(); }, - onSelectionChanged: () => this.selectionService.onSelectionChanged(this.gridOptions.api.getSelectedNodes()), + onSelectionChanged: () => this.selectionService.onSelectionChanged(this.agGrid.api.getSelectedNodes()), onBodyScroll: () => this.contextMenuService.closePostContextMenu(), - onRowDataUpdated: () => this.postsService.setCount(this.gridOptions.api.getDisplayedRowCount()), - onRowDataChanged: () => this.postsService.setCount(this.gridOptions.api.getDisplayedRowCount()), + onRowDataUpdated: () => this.postsService.setCount(this.agGrid.api.getDisplayedRowCount()), }; } diff --git a/vripper-ui/src/app/posts/posts.datasource.ts b/vripper-web-ui/src/app/posts/posts.datasource.ts similarity index 63% rename from vripper-ui/src/app/posts/posts.datasource.ts rename to vripper-web-ui/src/app/posts/posts.datasource.ts index 51e88e5c..dd5414c8 100644 --- a/vripper-ui/src/app/posts/posts.datasource.ts +++ b/vripper-web-ui/src/app/posts/posts.datasource.ts @@ -1,42 +1,42 @@ import {Post} from '../domain/post-state.model'; import {Subscription} from 'rxjs'; import {WsConnectionService} from '../services/ws-connection.service'; -import {GridOptions, RowNode} from 'ag-grid-community'; +import {GridApi, IRowNode} from 'ag-grid-community'; import {NgZone} from '@angular/core'; export class PostsDataSource { subscriptions: Subscription[] = []; - constructor(private ws: WsConnectionService, private gridOptions: GridOptions, private zone: NgZone) { + constructor(private ws: WsConnectionService, private gridApi: GridApi, private zone: NgZone) { } connect() { this.subscriptions.push(this.ws.posts$.subscribe((e: Post[]) => { this.zone.run(() => { - const toAdd = []; - const toUpdate = []; + const toAdd: Post[] = []; + const toUpdate: Post[] = []; e.forEach(v => { - if (this.gridOptions.api.getRowNode(v.postId) == null) { + if (this.gridApi.getRowNode(v.postId) == null) { toAdd.push(v); } else { toUpdate.push(v); } }); - this.gridOptions.api.applyTransaction({update: toUpdate, add: toAdd}); + this.gridApi.applyTransaction({update: toUpdate, add: toAdd}); }); })); this.subscriptions.push(this.ws.postsRemove$.subscribe((e: string[]) => { this.zone.run(() => { - const toRemove = []; + const toRemove: any[] = []; e.forEach(v => { - const rowNode: RowNode = this.gridOptions.api.getRowNode(v); + const rowNode: IRowNode | undefined = this.gridApi.getRowNode(v); if (rowNode != null) { toRemove.push(rowNode.data); } return; }); - this.gridOptions.api.applyTransaction({remove: toRemove}); + this.gridApi.applyTransaction({remove: toRemove}); }); })); } diff --git a/vripper-ui/src/app/preview-tooltip/preview-tooltip.component.ts b/vripper-web-ui/src/app/preview-tooltip/preview-tooltip.component.ts similarity index 100% rename from vripper-ui/src/app/preview-tooltip/preview-tooltip.component.ts rename to vripper-web-ui/src/app/preview-tooltip/preview-tooltip.component.ts diff --git a/vripper-ui/src/app/preview-tooltip/preview-tooltip.directive.ts b/vripper-web-ui/src/app/preview-tooltip/preview-tooltip.directive.ts similarity index 96% rename from vripper-ui/src/app/preview-tooltip/preview-tooltip.directive.ts rename to vripper-web-ui/src/app/preview-tooltip/preview-tooltip.directive.ts index b22ac7a9..8b016e85 100644 --- a/vripper-ui/src/app/preview-tooltip/preview-tooltip.directive.ts +++ b/vripper-web-ui/src/app/preview-tooltip/preview-tooltip.directive.ts @@ -6,8 +6,8 @@ import {ComponentPortal} from '@angular/cdk/portal'; @Directive({selector: '[appPreview]'}) export class AppPreviewDirective implements OnInit, OnDestroy { tooltipPortal = new ComponentPortal(AppPreviewComponent); - @Input() appPreview: string[]; - private overlayRef: OverlayRef; + @Input() appPreview!: string[]; + private overlayRef!: OverlayRef; constructor( private overlayPositionBuilder: OverlayPositionBuilder, diff --git a/vripper-ui/src/app/scan/scan.component.html b/vripper-web-ui/src/app/scan/scan.component.html similarity index 100% rename from vripper-ui/src/app/scan/scan.component.html rename to vripper-web-ui/src/app/scan/scan.component.html diff --git a/vripper-ui/src/app/scan/scan.component.scss b/vripper-web-ui/src/app/scan/scan.component.scss similarity index 100% rename from vripper-ui/src/app/scan/scan.component.scss rename to vripper-web-ui/src/app/scan/scan.component.scss diff --git a/vripper-ui/src/app/scan/scan.component.ts b/vripper-web-ui/src/app/scan/scan.component.ts similarity index 92% rename from vripper-ui/src/app/scan/scan.component.ts rename to vripper-web-ui/src/app/scan/scan.component.ts index 90d717ad..74fbc465 100644 --- a/vripper-ui/src/app/scan/scan.component.ts +++ b/vripper-web-ui/src/app/scan/scan.component.ts @@ -13,7 +13,7 @@ import {BehaviorSubject, Subject} from 'rxjs'; changeDetection: ChangeDetectionStrategy.OnPush }) export class ScanComponent implements OnInit, AfterViewInit { - input: string; + input!: string; loading: Subject = new BehaviorSubject(false); constructor( @@ -43,7 +43,7 @@ export class ScanComponent implements OnInit, AfterViewInit { processUrl(url: string) { this.loading.next(true); this.httpClient - .post<{ threadId: string; postId: string }>(this.serverService.baseUrl + '/post', {url: url}) + .post(this.serverService.baseUrl + '/post', {url: url}) .pipe( finalize(() => { this.close(); @@ -53,7 +53,7 @@ export class ScanComponent implements OnInit, AfterViewInit { .subscribe(response => { }, error => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { + this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', undefined, { duration: 5000 }); }); diff --git a/vripper-ui/src/app/services/app.service.ts b/vripper-web-ui/src/app/services/app.service.ts similarity index 97% rename from vripper-ui/src/app/services/app.service.ts rename to vripper-web-ui/src/app/services/app.service.ts index db980ff6..67f863ac 100644 --- a/vripper-ui/src/app/services/app.service.ts +++ b/vripper-web-ui/src/app/services/app.service.ts @@ -32,13 +32,13 @@ export class AppService { return this._darkTheme; } - private _settings: Settings; + private _settings!: Settings; get settings(): Settings { return {...this._settings}; } - private _renderer: Renderer2; + private _renderer!: Renderer2; set renderer(renderer: Renderer2) { this._renderer = renderer; diff --git a/vripper-ui/src/app/services/event-log.service.ts b/vripper-web-ui/src/app/services/event-log.service.ts similarity index 95% rename from vripper-ui/src/app/services/event-log.service.ts rename to vripper-web-ui/src/app/services/event-log.service.ts index d97824a1..e9c7502a 100644 --- a/vripper-ui/src/app/services/event-log.service.ts +++ b/vripper-web-ui/src/app/services/event-log.service.ts @@ -8,7 +8,7 @@ import {EventLogMessageDialogComponent} from '../event-log/message-dialog/event- providedIn: 'root' }) export class EventLogService { - private api: GridApi; + private api!: GridApi; private _count$: Subject = new BehaviorSubject(0); @@ -27,7 +27,7 @@ export class EventLogService { this.api = api; } - search(event) { + search(event: any) { if (this.api) { this.api.setQuickFilter(event); } diff --git a/vripper-ui/src/app/services/home-tabs.service.ts b/vripper-web-ui/src/app/services/home-tabs.service.ts similarity index 100% rename from vripper-ui/src/app/services/home-tabs.service.ts rename to vripper-web-ui/src/app/services/home-tabs.service.ts diff --git a/vripper-ui/src/app/services/link-collector.service.ts b/vripper-web-ui/src/app/services/link-collector.service.ts similarity index 92% rename from vripper-ui/src/app/services/link-collector.service.ts rename to vripper-web-ui/src/app/services/link-collector.service.ts index 11c2a258..f05c9a6e 100644 --- a/vripper-ui/src/app/services/link-collector.service.ts +++ b/vripper-web-ui/src/app/services/link-collector.service.ts @@ -6,7 +6,7 @@ import {GridApi} from 'ag-grid-community'; providedIn: 'root' }) export class LinkCollectorService { - private api: GridApi; + private api!: GridApi; private _count$: Subject = new BehaviorSubject(0); @@ -18,7 +18,7 @@ export class LinkCollectorService { this.api = api; } - search(event) { + search(event: any) { if (this.api) { this.api.setQuickFilter(event); } diff --git a/vripper-ui/src/app/services/multi-post.service.ts b/vripper-web-ui/src/app/services/multi-post.service.ts similarity index 100% rename from vripper-ui/src/app/services/multi-post.service.ts rename to vripper-web-ui/src/app/services/multi-post.service.ts diff --git a/vripper-ui/src/app/services/post-context-menu.service.ts b/vripper-web-ui/src/app/services/post-context-menu.service.ts similarity index 95% rename from vripper-ui/src/app/services/post-context-menu.service.ts rename to vripper-web-ui/src/app/services/post-context-menu.service.ts index ad77ba66..b8912a39 100644 --- a/vripper-ui/src/app/services/post-context-menu.service.ts +++ b/vripper-web-ui/src/app/services/post-context-menu.service.ts @@ -12,8 +12,8 @@ import {filter, take} from 'rxjs/operators'; export class PostContextMenuService { postContextMenuPortal = new ComponentPortal(PostContextMenuComponent); - postContext: Subscription; - private postContextMenuOverlayRef: OverlayRef; + postContext!: Subscription; + private postContextMenuOverlayRef: OverlayRef | null = null; constructor(private overlayPositionBuilder: OverlayPositionBuilder, private overlay: Overlay) { } diff --git a/vripper-ui/src/app/services/posts.service.ts b/vripper-web-ui/src/app/services/posts.service.ts similarity index 80% rename from vripper-ui/src/app/services/posts.service.ts rename to vripper-web-ui/src/app/services/posts.service.ts index c64f5579..9b245a6c 100644 --- a/vripper-ui/src/app/services/posts.service.ts +++ b/vripper-web-ui/src/app/services/posts.service.ts @@ -6,7 +6,7 @@ import {BehaviorSubject, Observable, Subject} from 'rxjs'; providedIn: 'root' }) export class PostsService { - private api: GridApi; + private api!: GridApi; private _count$: Subject = new BehaviorSubject(0); @@ -22,10 +22,10 @@ export class PostsService { this.api = api; } - public remove(toRemove: { postId: string }[]) { - const removeTx = []; + public remove(toRemove: string[]) { + const removeTx: any[] = []; toRemove.forEach(element => { - const nodeToDelete = this.api.getRowNode(element.postId); + const nodeToDelete = this.api.getRowNode(element); if (nodeToDelete != null) { removeTx.push(nodeToDelete.data); } @@ -33,7 +33,7 @@ export class PostsService { this.api.applyTransaction({remove: removeTx}); } - search(event) { + search(event: any) { if (this.api) { this.api.setQuickFilter(event); } diff --git a/vripper-ui/src/app/services/selection-service.ts b/vripper-web-ui/src/app/services/selection-service.ts similarity index 55% rename from vripper-ui/src/app/services/selection-service.ts rename to vripper-web-ui/src/app/services/selection-service.ts index 9323606f..c936fc91 100644 --- a/vripper-ui/src/app/services/selection-service.ts +++ b/vripper-web-ui/src/app/services/selection-service.ts @@ -1,19 +1,19 @@ import {Injectable} from '@angular/core'; import {Observable, Subject} from 'rxjs'; -import {RowNode} from 'ag-grid-community'; +import {IRowNode} from 'ag-grid-community'; @Injectable({ providedIn: 'root' }) export class SelectionService { - private _selected$: Subject = new Subject(); + private _selected$: Subject = new Subject(); - get selected$(): Observable { + get selected$(): Observable { return this._selected$.asObservable(); } - public onSelectionChanged(selected: RowNode[]) { + public onSelectionChanged(selected: IRowNode[]) { this._selected$.next(selected); } } diff --git a/vripper-ui/src/app/services/server-service.ts b/vripper-web-ui/src/app/services/server-service.ts similarity index 86% rename from vripper-ui/src/app/services/server-service.ts rename to vripper-web-ui/src/app/services/server-service.ts index 85a5c5cc..7229ca2e 100644 --- a/vripper-ui/src/app/services/server-service.ts +++ b/vripper-web-ui/src/app/services/server-service.ts @@ -8,7 +8,7 @@ export class ServerService { constructor() { } - private _baseUrl: string; + private _baseUrl!: string; get baseUrl(): string { return this._baseUrl; @@ -18,7 +18,7 @@ export class ServerService { this._baseUrl = baseUrl; } - private _wsBaseUrl: string; + private _wsBaseUrl!: string; get wsBaseUrl(): string { return this._wsBaseUrl; diff --git a/vripper-ui/src/app/services/ws-connection.service.ts b/vripper-web-ui/src/app/services/ws-connection.service.ts similarity index 70% rename from vripper-ui/src/app/services/ws-connection.service.ts rename to vripper-web-ui/src/app/services/ws-connection.service.ts index c9c38e04..084c0396 100644 --- a/vripper-ui/src/app/services/ws-connection.service.ts +++ b/vripper-web-ui/src/app/services/ws-connection.service.ts @@ -1,40 +1,35 @@ import {Injectable} from '@angular/core'; -import {BehaviorSubject, Observable, Subject, Subscription} from 'rxjs'; +import {BehaviorSubject, Observable, Subscription} from 'rxjs'; import {environment} from 'src/environments/environment'; import {ServerService} from './server-service'; import {MultiPostModel} from '../domain/multi-post.model'; import {WSMessage} from '../domain/ws-message.model'; import {Photo} from '../domain/photo-model'; import {Post} from '../domain/post-state.model'; -import {LoggedUser} from '../domain/logged-user.model'; -import {DownloadSpeed} from '../domain/download-speed.model'; import {GlobalState} from '../domain/global-state.model'; import {map, share} from 'rxjs/operators'; import {RxStomp, RxStompConfig, RxStompState} from '@stomp/rx-stomp'; -import {ElectronService} from 'ngx-electron'; import {EventLog} from '../domain/event.model'; @Injectable({ providedIn: 'root' }) export class WsConnectionService { - rxStomp: RxStomp; - private globalState: Observable; - private speed: Observable; - private user: Observable; - private posts: Observable>; - private events: Observable>; - private postDetails: Observable>; - private multiPostModels: Observable>; - private queuedRemove: Observable>; - private postsRemove: Observable>; - private eventsRemove: Observable>; - private postIdDetails: string; - - private connectionState$: Subject = new BehaviorSubject(RxStompState.CLOSED); - private stateSubscription: Subscription; - - constructor(private serverService: ServerService, private electronService: ElectronService) { + rxStomp!: RxStomp; + private globalState!: Observable; + private posts!: Observable>; + private events!: Observable>; + private postDetails!: Observable>; + private multiPostModels!: Observable>; + private queuedRemove!: Observable>; + private postsRemove!: Observable>; + private eventsRemove!: Observable>; + private postIdDetails!: string; + + private connectionState$ = new BehaviorSubject(RxStompState.CLOSED); + private stateSubscription!: Subscription; + + constructor(private serverService: ServerService) { this.init(); } @@ -46,14 +41,6 @@ export class WsConnectionService { return this.globalState; } - get speed$(): Observable { - return this.speed; - } - - get user$(): Observable { - return this.user; - } - get posts$(): Observable { return this.posts; } @@ -79,26 +66,10 @@ export class WsConnectionService { } init() { - if (this.electronService.isElectronApp) { - const portRequest = setInterval(() => { - this.electronService.ipcRenderer.send('get-port'); - }, 200); - - // wait for MainIPC - this.electronService.ipcRenderer.once('port', (event, port) => { - clearInterval(portRequest); - console.log('server running on port', port); - this.serverService.baseUrl = 'http://localhost:' + port; - this.serverService.wsBaseUrl = 'ws://localhost:' + port; - this.connect(); - this.subscribe(); - }); - } else { this.serverService.baseUrl = environment.localhost; this.serverService.wsBaseUrl = environment.ws; this.connect(); this.subscribe(); - } } connect() { @@ -146,7 +117,7 @@ export class WsConnectionService { share() ); - this.globalState = this.rxStomp.watch('/topic/download-state').pipe( + this.globalState = this.rxStomp.watch('/topic/state').pipe( map(e => { const state: GlobalState = JSON.parse(e.body); return state; @@ -154,20 +125,6 @@ export class WsConnectionService { share() ); - this.speed = this.rxStomp.watch('/topic/speed').pipe( - map(e => { - return JSON.parse(e.body); - }), - share() - ); - - this.user = this.rxStomp.watch('/topic/user').pipe( - map(e => { - return JSON.parse(e.body); - }), - share() - ); - this.posts = this.rxStomp.watch('/topic/posts').pipe( map(e => { const posts: Array = []; @@ -175,16 +132,13 @@ export class WsConnectionService { posts.push( new Post( element.postId, - element.title, + element.postTitle, element.done === 0 && element.total === 0 ? 0 : (element.done / element.total) * 100, element.status, - element.removed, element.url, element.done, element.total, element.hosts, - element.thanked, - element.previews, element.metadata, element.addedOn, element.rank + 1 diff --git a/vripper-ui/src/app/settings/settings.component.html b/vripper-web-ui/src/app/settings/settings.component.html similarity index 79% rename from vripper-ui/src/app/settings/settings.component.html rename to vripper-web-ui/src/app/settings/settings.component.html index 62e6a0c0..70d1d581 100644 --- a/vripper-ui/src/app/settings/settings.component.html +++ b/vripper-web-ui/src/app/settings/settings.component.html @@ -20,9 +20,6 @@

Preferences

required /> -
@@ -35,7 +32,7 @@

Preferences

- Save posts in sub folders Preferences Preferences
Select a proxy - + {{mirror}} - + Enable ViperGirls Authentication -
+
- +
- Send a like to poster + Send a like to poster
-
- Send a like - when download starts - - -
@@ -199,17 +188,6 @@

Preferences

- -
-
- Monitor Clipboard - - -
-
-
diff --git a/vripper-ui/src/app/settings/settings.component.scss b/vripper-web-ui/src/app/settings/settings.component.scss similarity index 100% rename from vripper-ui/src/app/settings/settings.component.scss rename to vripper-web-ui/src/app/settings/settings.component.scss diff --git a/vripper-ui/src/app/settings/settings.component.ts b/vripper-web-ui/src/app/settings/settings.component.ts similarity index 56% rename from vripper-ui/src/app/settings/settings.component.ts rename to vripper-web-ui/src/app/settings/settings.component.ts index 2ac0c0b3..8010a08c 100644 --- a/vripper-ui/src/app/settings/settings.component.ts +++ b/vripper-web-ui/src/app/settings/settings.component.ts @@ -1,15 +1,12 @@ import {finalize} from 'rxjs/operators'; import {AppService} from '../services/app.service'; -import {ClipboardService} from '../services/clipboard.service'; import {ChangeDetectionStrategy, Component, NgZone, OnDestroy, OnInit} from '@angular/core'; import {HttpClient} from '@angular/common/http'; import {FormControl, FormGroup} from '@angular/forms'; import {MatSnackBar} from '@angular/material/snack-bar'; import {ServerService} from '../services/server-service'; -import {ElectronService} from 'ngx-electron'; -import {Settings} from '../domain/settings.model'; +import {ConnectionSettings, DownloadSettings, Settings, ViperSettings} from '../domain/settings.model'; import {EMPTY, Observable, Subscription} from 'rxjs'; -import OpenDialogReturnValue = Electron.OpenDialogReturnValue; @Component({ selector: 'app-settings', @@ -26,29 +23,28 @@ export class SettingsComponent implements OnInit, OnDestroy { subscriptions: Subscription[] = []; viperGirlsSettingsForm = new FormGroup({ - vLogin: new FormControl(false), - vUsername: new FormControl(''), - vPassword: new FormControl(''), - vThanks: new FormControl(false), - leaveThanksOnStart: new FormControl(false), - vProxy: new FormControl('') + login: new FormControl(false), + username: new FormControl(''), + password: new FormControl(''), + thanks: new FormControl(false), + host: new FormControl('') }); downloadSettingsForm = new FormGroup({ downloadPath: new FormControl(''), autoStart: new FormControl(false), forceOrder: new FormControl(false), - subLocation: new FormControl(false), + forumSubfolder: new FormControl(false), threadSubLocation: new FormControl(false), clearCompleted: new FormControl(false), appendPostId: new FormControl(false) }); connectionSettingsForm = new FormGroup({ - maxThreads: new FormControl(''), - maxTotalThreads: new FormControl(''), - connectionTimeout: new FormControl(''), - maxAttempts: new FormControl('') + maxThreads: new FormControl(0), + maxTotalThreads: new FormControl(0), + timeout: new FormControl(0), + maxAttempts: new FormControl(0) }); desktopSettingsForm = new FormGroup({ @@ -56,27 +52,16 @@ export class SettingsComponent implements OnInit, OnDestroy { }); eventLogSettingsForm = new FormGroup({ - maxEventLog: new FormControl('') + maxEventLog: new FormControl(0) }); constructor( private httpClient: HttpClient, private _snackBar: MatSnackBar, private serverService: ServerService, - public electronService: ElectronService, - private clipboardService: ClipboardService, private appService: AppService, private zone: NgZone ) { - this.subscriptions.push(this.viperGirlsSettingsForm.get('vThanks').valueChanges.subscribe(c => { - const formControl: FormControl = this.viperGirlsSettingsForm.get('leaveThanksOnStart') as FormControl; - if (!c) { - formControl.setValue(false); - formControl.disable(); - } else { - formControl.enable(); - } - })); } updateTheme() { @@ -92,13 +77,13 @@ export class SettingsComponent implements OnInit, OnDestroy { this.mirrors = this.httpClient.get(this.serverService.baseUrl + '/settings/proxies'); this.httpClient.get(this.serverService.baseUrl + '/settings') .subscribe(data => { - this.viperGirlsSettingsForm.reset(data); - this.downloadSettingsForm.reset(data); - this.connectionSettingsForm.reset(data); + this.viperGirlsSettingsForm.reset(data.viperSettings); + this.downloadSettingsForm.reset(data.downloadSettings); + this.connectionSettingsForm.reset(data.connectionSettings); this.desktopSettingsForm.reset(data); this.eventLogSettingsForm.reset(data); }, error => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { + this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', undefined, { duration: 5000 }); }); @@ -108,49 +93,32 @@ export class SettingsComponent implements OnInit, OnDestroy { this.subscriptions.forEach(s => s.unsubscribe()); } - browse() { - this.electronService.remote.dialog - .showOpenDialog(this.electronService.remote.getCurrentWindow(), { - properties: ['openDirectory'] - }) - .then((value: OpenDialogReturnValue) => { - this.zone.run(() => { - if (!value.canceled && value.filePaths !== undefined) { - this.downloadSettingsForm.get('downloadPath').setValue(value.filePaths[0]); - this.downloadSettingsForm.get('downloadPath').markAsDirty(); - this.downloadSettingsForm.get('downloadPath').markAsTouched(); - } - }); - }); - } - onSubmit(): void { this.loading = true; this.httpClient .post(this.serverService.baseUrl + '/settings', { - ...this.viperGirlsSettingsForm.value, - ...this.downloadSettingsForm.value, - ...this.connectionSettingsForm.value, + viperSettings: {...this.viperGirlsSettingsForm.value as ViperSettings}, + connectionSettings: {...this.connectionSettingsForm.value as ConnectionSettings}, + downloadSettings: {...this.downloadSettingsForm.value as DownloadSettings}, darkTheme: this.darkTheme, ...this.desktopSettingsForm.value, ...this.eventLogSettingsForm.value - }) + } as Settings) .pipe(finalize(() => (this.loading = false))) .subscribe( data => { - this._snackBar.open('Settings updated', null, { + this._snackBar.open('Settings updated', undefined, { duration: 5000 }); - this.viperGirlsSettingsForm.reset(data); - this.downloadSettingsForm.reset(data); - this.connectionSettingsForm.reset(data); + this.viperGirlsSettingsForm.reset(data.viperSettings); + this.downloadSettingsForm.reset(data.downloadSettings); + this.connectionSettingsForm.reset(data.connectionSettings); this.desktopSettingsForm.reset(data); this.eventLogSettingsForm.reset(data); - this.clipboardService.init(data); this.updateSettings(data); }, error => { - this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', null, { + this._snackBar.open(error?.error?.message || 'Unexpected error, check log file', undefined, { duration: 5000 }); } diff --git a/vripper-ui/src/app/status-bar/status-bar.component.html b/vripper-web-ui/src/app/status-bar/status-bar.component.html similarity index 82% rename from vripper-ui/src/app/status-bar/status-bar.component.html rename to vripper-web-ui/src/app/status-bar/status-bar.component.html index 6508c8b2..d9888b56 100644 --- a/vripper-ui/src/app/status-bar/status-bar.component.html +++ b/vripper-web-ui/src/app/status-bar/status-bar.component.html @@ -1,8 +1,8 @@
Selected: {{length}} - {{ speed.speed + '/s' }} + {{ state.downloadSpeed + '/s' }} Downloading: {{ state!.running }} Pending: {{ state!.remaining }} Error: {{ state!.error }} diff --git a/vripper-ui/src/app/status-bar/status-bar.component.scss b/vripper-web-ui/src/app/status-bar/status-bar.component.scss similarity index 100% rename from vripper-ui/src/app/status-bar/status-bar.component.scss rename to vripper-web-ui/src/app/status-bar/status-bar.component.scss diff --git a/vripper-web-ui/src/app/status-bar/status-bar.component.scss-theme.scss b/vripper-web-ui/src/app/status-bar/status-bar.component.scss-theme.scss new file mode 100644 index 00000000..0e375cf4 --- /dev/null +++ b/vripper-web-ui/src/app/status-bar/status-bar.component.scss-theme.scss @@ -0,0 +1,12 @@ +@use 'sass:map'; +@use '@angular/material' as mat; + +@mixin status-bar-component-theme($theme) { + + $theme-color: mat.get-color-config($theme); + $primary: map.get($theme-color, 'primary'); + + .status-bar { + color: mat.get-color-from-palette($primary, '900-contrast'); + } +} diff --git a/vripper-ui/src/app/status-bar/status-bar.component.ts b/vripper-web-ui/src/app/status-bar/status-bar.component.ts similarity index 81% rename from vripper-ui/src/app/status-bar/status-bar.component.ts rename to vripper-web-ui/src/app/status-bar/status-bar.component.ts index 357a81f9..f1f860df 100644 --- a/vripper-ui/src/app/status-bar/status-bar.component.ts +++ b/vripper-web-ui/src/app/status-bar/status-bar.component.ts @@ -1,6 +1,5 @@ import {SelectionService} from '../services/selection-service'; import {ChangeDetectionStrategy, Component, NgZone, OnDestroy, OnInit} from '@angular/core'; -import {DownloadSpeed} from '../domain/download-speed.model'; import {WsConnectionService} from '../services/ws-connection.service'; import {BehaviorSubject, Subject, Subscription} from 'rxjs'; import {GlobalState} from '../domain/global-state.model'; @@ -12,8 +11,7 @@ import {GlobalState} from '../domain/global-state.model'; changeDetection: ChangeDetectionStrategy.OnPush }) export class StatusBarComponent implements OnInit, OnDestroy { - downloadSpeed$: Subject = new BehaviorSubject(new DownloadSpeed('0 B')); - globalState$: Subject = new BehaviorSubject(new GlobalState(0, 0, 0)); + globalState$: Subject = new BehaviorSubject(new GlobalState(0, 0, 0, '', '')); selected$: Subject = new BehaviorSubject(0); subscriptions: Subscription[] = []; @@ -21,7 +19,6 @@ export class StatusBarComponent implements OnInit, OnDestroy { } ngOnInit() { - this.subscriptions.push(this.ws.speed$.subscribe(e => this.ngZone.run(() => this.downloadSpeed$.next(e)))); this.subscriptions.push(this.ws.globalState$.subscribe(e => this.ngZone.run(() => this.globalState$.next(e)))); this.subscriptions.push(this.selectionService.selected$.subscribe(selected => this.ngZone.run(() => this.selected$.next(selected.length)))); } diff --git a/vripper-ui/src/app/toolbar/toolbar.component.html b/vripper-web-ui/src/app/toolbar/toolbar.component.html similarity index 90% rename from vripper-ui/src/app/toolbar/toolbar.component.html rename to vripper-web-ui/src/app/toolbar/toolbar.component.html index 054c409e..e1431ff9 100644 --- a/vripper-ui/src/app/toolbar/toolbar.component.html +++ b/vripper-web-ui/src/app/toolbar/toolbar.component.html @@ -17,11 +17,6 @@ matTooltip="Stop selected"> pause -
@@ -54,7 +49,7 @@

- Logged in as: {{ user.user }} + Logged in as: {{ user }}