From 6a63694a3e2878da95bea2376f267061a2e60ac1 Mon Sep 17 00:00:00 2001 From: Eike Kettner Date: Wed, 10 Mar 2021 19:48:56 +0100 Subject: [PATCH 1/6] Convert unit tests to munit --- build.sbt | 38 +++--- .../StanfordTextClassifierSuite.scala | 4 +- .../contact/ContactAnnotateSpec.scala | 4 +- .../docspell/analysis/date/DateFindSpec.scala | 4 +- .../analysis/nlp/BaseCRFAnnotatorSuite.scala | 12 +- .../nlp/StanfordNerAnnotatorSuite.scala | 16 +-- .../analysis/split/TestSplitterSpec.scala | 4 +- .../scala/docspell/common/FileNameTest.scala | 4 +- .../test/scala/docspell/common/GlobTest.scala | 4 +- .../common/MetaProposalListTest.scala | 4 +- .../docspell/common/NerLabelSpanTest.scala | 4 +- .../docspell/convert/ConversionTest.scala | 87 +++++++------- .../convert/extern/ExternConvTest.scala | 113 +++++++++--------- .../extract/ocr/TextExtractionSuite.scala | 10 +- .../docspell/extract/odf/OdfExtractTest.scala | 4 +- .../extract/pdfbox/PdfMetaDataTest.scala | 4 +- .../extract/pdfbox/PdfboxExtractTest.scala | 4 +- .../extract/pdfbox/PdfboxPreviewTest.scala | 4 +- .../docspell/extract/poi/PoiExtractTest.scala | 4 +- .../docspell/extract/rtf/RtfExtractTest.scala | 4 +- .../scala/docspell/files/ImageSizeTest.scala | 4 +- .../test/scala/docspell/files/ZipTest.scala | 4 +- .../docspell/joex/analysis/NerFileTest.scala | 4 +- .../joex/scheduler/CountingSchemeSpec.scala | 4 +- .../generator/ItemQueryGeneratorTest.scala | 4 +- .../docspell/store/qb/QueryBuilderTest.scala | 4 +- .../store/qb/impl/ConditionBuilderTest.scala | 4 +- .../docspell/store/qb/impl/DSLTest.scala | 4 +- .../store/qb/impl/SelectBuilderTest.scala | 4 +- .../store/queries/QueryWildcardTest.scala | 4 +- 30 files changed, 175 insertions(+), 197 deletions(-) diff --git a/build.sbt b/build.sbt index 3a1ca00a..290f5109 100644 --- a/build.sbt +++ b/build.sbt @@ -45,12 +45,6 @@ val sharedSettings = Seq( (scalacOptions.value.filter(o => !o.contains("-Xlint") && !o.contains("-W"))) ) ++ scalafixSettings -val testSettings = Seq( - testFrameworks += new TestFramework("minitest.runner.Framework"), - libraryDependencies ++= Dependencies.miniTest ++ Dependencies.logging.map(_ % Test), - Test / fork := true -) - val testSettingsMUnit = Seq( libraryDependencies ++= Dependencies.munit.map(_ % Test), testFrameworks += new TestFramework("munit.Framework") @@ -231,7 +225,7 @@ val common = project .in(file("modules/common")) .disablePlugins(RevolverPlugin) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings( name := "docspell-common", libraryDependencies ++= @@ -249,7 +243,7 @@ val files = project .in(file("modules/files")) .disablePlugins(RevolverPlugin) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings( name := "docspell-files", libraryDependencies ++= @@ -308,7 +302,7 @@ val store = project .in(file("modules/store")) .disablePlugins(RevolverPlugin) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings( name := "docspell-store", libraryDependencies ++= @@ -330,7 +324,7 @@ val extract = project .in(file("modules/extract")) .disablePlugins(RevolverPlugin) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings( name := "docspell-extract", libraryDependencies ++= @@ -347,7 +341,7 @@ val convert = project .in(file("modules/convert")) .disablePlugins(RevolverPlugin) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings( name := "docspell-convert", libraryDependencies ++= @@ -361,7 +355,7 @@ val analysis = project .disablePlugins(RevolverPlugin) .enablePlugins(NerModelsPlugin) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings(NerModelsPlugin.nerClassifierSettings) .settings( name := "docspell-analysis", @@ -375,7 +369,7 @@ val ftsclient = project .in(file("modules/fts-client")) .disablePlugins(RevolverPlugin) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings( name := "docspell-fts-client", libraryDependencies ++= Seq.empty @@ -386,7 +380,7 @@ val ftssolr = project .in(file("modules/fts-solr")) .disablePlugins(RevolverPlugin) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings( name := "docspell-fts-solr", libraryDependencies ++= @@ -402,7 +396,7 @@ val restapi = project .disablePlugins(RevolverPlugin) .enablePlugins(OpenApiSchema) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings(openapiScalaSettings) .settings( name := "docspell-restapi", @@ -420,7 +414,7 @@ val joexapi = project .disablePlugins(RevolverPlugin) .enablePlugins(OpenApiSchema) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings(openapiScalaSettings) .settings( name := "docspell-joexapi", @@ -438,7 +432,7 @@ val backend = project .in(file("modules/backend")) .disablePlugins(RevolverPlugin) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings( name := "docspell-backend", libraryDependencies ++= @@ -473,7 +467,7 @@ val joex = project .in(file("modules/joex")) .enablePlugins(BuildInfoPlugin, JavaServerAppPackaging, DebianPlugin, SystemdPlugin) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings(debianSettings("docspell-joex")) .settings(buildInfoSettings) .settings( @@ -512,7 +506,7 @@ val restserver = project .in(file("modules/restserver")) .enablePlugins(BuildInfoPlugin, JavaServerAppPackaging, DebianPlugin, SystemdPlugin) .settings(sharedSettings) - .settings(testSettings) + .settings(testSettingsMUnit) .settings(debianSettings("docspell-server")) .settings(buildInfoSettings) .settings( @@ -740,7 +734,8 @@ def packageTools(logger: Logger, dir: File, version: String): Seq[File] = { wx / "icons" / "logo-96.png" -> "icons/logo-96.png", wx / "manifest.json" -> "manifest.json" ), - webext + webext, + None ) val excludes = Seq(wx, target) @@ -757,7 +752,8 @@ def packageTools(logger: Logger, dir: File, version: String): Seq[File] = { wx / "native/app_manifest.json" -> s"docspell-tools-${version}/firefox/native/app_manifest.json", wx / "native/native.py" -> s"docspell-tools-${version}/firefox/native/native.py" ) ++ files, - archive + archive, + None ) Seq(archive) diff --git a/modules/analysis/src/test/scala/docspell/analysis/classifier/StanfordTextClassifierSuite.scala b/modules/analysis/src/test/scala/docspell/analysis/classifier/StanfordTextClassifierSuite.scala index 0229585c..1fcf5e81 100644 --- a/modules/analysis/src/test/scala/docspell/analysis/classifier/StanfordTextClassifierSuite.scala +++ b/modules/analysis/src/test/scala/docspell/analysis/classifier/StanfordTextClassifierSuite.scala @@ -1,6 +1,6 @@ package docspell.analysis.classifier -import minitest._ +import munit._ import cats.effect._ import scala.concurrent.ExecutionContext import java.nio.file.Paths @@ -10,7 +10,7 @@ import fs2.Stream import cats.data.Kleisli import TextClassifier.Data -object StanfordTextClassifierSuite extends SimpleTestSuite { +class StanfordTextClassifierSuite extends FunSuite { val logger = Logger.log4s[IO](org.log4s.getLogger) implicit val CS = IO.contextShift(ExecutionContext.global) diff --git a/modules/analysis/src/test/scala/docspell/analysis/contact/ContactAnnotateSpec.scala b/modules/analysis/src/test/scala/docspell/analysis/contact/ContactAnnotateSpec.scala index aaff9203..106dad9e 100644 --- a/modules/analysis/src/test/scala/docspell/analysis/contact/ContactAnnotateSpec.scala +++ b/modules/analysis/src/test/scala/docspell/analysis/contact/ContactAnnotateSpec.scala @@ -1,9 +1,9 @@ package docspell.analysis.contact import docspell.common.{NerLabel, NerTag} -import minitest.SimpleTestSuite +import munit._ -object ContactAnnotateSpec extends SimpleTestSuite { +class ContactAnnotateSpec extends FunSuite { test("find email") { diff --git a/modules/analysis/src/test/scala/docspell/analysis/date/DateFindSpec.scala b/modules/analysis/src/test/scala/docspell/analysis/date/DateFindSpec.scala index 705d0f43..3711434f 100644 --- a/modules/analysis/src/test/scala/docspell/analysis/date/DateFindSpec.scala +++ b/modules/analysis/src/test/scala/docspell/analysis/date/DateFindSpec.scala @@ -1,11 +1,11 @@ package docspell.analysis.date import docspell.files.TestFiles -import minitest.SimpleTestSuite +import munit._ import docspell.common._ import java.time._ -object DateFindSpec extends SimpleTestSuite { +class DateFindSpec extends FunSuite { test("find simple dates") { val expect = Vector( diff --git a/modules/analysis/src/test/scala/docspell/analysis/nlp/BaseCRFAnnotatorSuite.scala b/modules/analysis/src/test/scala/docspell/analysis/nlp/BaseCRFAnnotatorSuite.scala index 29b3b966..340359e1 100644 --- a/modules/analysis/src/test/scala/docspell/analysis/nlp/BaseCRFAnnotatorSuite.scala +++ b/modules/analysis/src/test/scala/docspell/analysis/nlp/BaseCRFAnnotatorSuite.scala @@ -2,19 +2,17 @@ package docspell.analysis.nlp import docspell.analysis.Env import docspell.common.Language.NLPLanguage -import minitest.SimpleTestSuite +import munit._ import docspell.files.TestFiles import docspell.common._ -object BaseCRFAnnotatorSuite extends SimpleTestSuite { +class BaseCRFAnnotatorSuite extends FunSuite { def annotate(language: NLPLanguage): String => Vector[NerLabel] = BasicCRFAnnotator.nerAnnotate(BasicCRFAnnotator.Cache.getAnnotator(language)) test("find english ner labels") { - if (Env.isCI) { - ignore("Test ignored on travis.") - } + assume(!Env.isCI, "Test ignored on travis.") val labels = annotate(Language.English)(TestFiles.letterENText) val expect = Vector( @@ -52,9 +50,7 @@ object BaseCRFAnnotatorSuite extends SimpleTestSuite { } test("find german ner labels") { - if (Env.isCI) { - ignore("Test ignored on travis.") - } + assume(!Env.isCI, "Test ignored on travis.") val labels = annotate(Language.German)(TestFiles.letterDEText) val expect = Vector( diff --git a/modules/analysis/src/test/scala/docspell/analysis/nlp/StanfordNerAnnotatorSuite.scala b/modules/analysis/src/test/scala/docspell/analysis/nlp/StanfordNerAnnotatorSuite.scala index 91ab7a39..46bcb841 100644 --- a/modules/analysis/src/test/scala/docspell/analysis/nlp/StanfordNerAnnotatorSuite.scala +++ b/modules/analysis/src/test/scala/docspell/analysis/nlp/StanfordNerAnnotatorSuite.scala @@ -4,22 +4,20 @@ import java.nio.file.Paths import cats.effect.IO import docspell.analysis.Env -import minitest.SimpleTestSuite +import munit._ import docspell.files.TestFiles import docspell.common._ import docspell.common.syntax.FileSyntax._ import edu.stanford.nlp.pipeline.StanfordCoreNLP -object StanfordNerAnnotatorSuite extends SimpleTestSuite { +class StanfordNerAnnotatorSuite extends FunSuite { lazy val germanClassifier = new StanfordCoreNLP(Properties.nerGerman(None, false)) lazy val englishClassifier = new StanfordCoreNLP(Properties.nerEnglish(None)) test("find english ner labels") { - if (Env.isCI) { - ignore("Test ignored on travis.") - } + assume(!Env.isCI, "Test ignored on travis.") val labels = StanfordNerAnnotator.nerAnnotate(englishClassifier, TestFiles.letterENText) @@ -58,9 +56,7 @@ object StanfordNerAnnotatorSuite extends SimpleTestSuite { } test("find german ner labels") { - if (Env.isCI) { - ignore("Test ignored on travis.") - } + assume(!Env.isCI, "Test ignored on travis.") val labels = StanfordNerAnnotator.nerAnnotate(germanClassifier, TestFiles.letterDEText) @@ -83,9 +79,7 @@ object StanfordNerAnnotatorSuite extends SimpleTestSuite { } test("regexner-only annotator") { - if (Env.isCI) { - ignore("Test ignored on travis.") - } + assume(!Env.isCI, "Test ignored on travis.") val regexNerContent = s"""(?i)volantino ag${"\t"}ORGANIZATION${"\t"}LOCATION,PERSON,MISC${"\t"}3 diff --git a/modules/analysis/src/test/scala/docspell/analysis/split/TestSplitterSpec.scala b/modules/analysis/src/test/scala/docspell/analysis/split/TestSplitterSpec.scala index 4c17741d..81c37702 100644 --- a/modules/analysis/src/test/scala/docspell/analysis/split/TestSplitterSpec.scala +++ b/modules/analysis/src/test/scala/docspell/analysis/split/TestSplitterSpec.scala @@ -1,8 +1,8 @@ package docspell.analysis.split -import minitest.SimpleTestSuite +import munit._ -object TestSplitterSpec extends SimpleTestSuite { +class TestSplitterSpec extends FunSuite { test("simple splitting") { val text = """hiermit kündige ich meine Mitgliedschaft in der Kranken- und diff --git a/modules/common/src/test/scala/docspell/common/FileNameTest.scala b/modules/common/src/test/scala/docspell/common/FileNameTest.scala index 8b2778d7..05b12d74 100644 --- a/modules/common/src/test/scala/docspell/common/FileNameTest.scala +++ b/modules/common/src/test/scala/docspell/common/FileNameTest.scala @@ -1,8 +1,8 @@ package docspell.common -import minitest._ +import munit._ -object FileNameTest extends SimpleTestSuite { +class FileNameTest extends FunSuite { test("make filename") { val data = List( diff --git a/modules/common/src/test/scala/docspell/common/GlobTest.scala b/modules/common/src/test/scala/docspell/common/GlobTest.scala index 2e650174..be20b9ba 100644 --- a/modules/common/src/test/scala/docspell/common/GlobTest.scala +++ b/modules/common/src/test/scala/docspell/common/GlobTest.scala @@ -1,9 +1,9 @@ package docspell.common -import minitest._ +import munit._ import Glob._ -object GlobTest extends SimpleTestSuite { +class GlobTest extends FunSuite { test("literals") { assert(Glob.pattern(Pattern(Segment(Token.Literal("hello")))).matches(true)("hello")) diff --git a/modules/common/src/test/scala/docspell/common/MetaProposalListTest.scala b/modules/common/src/test/scala/docspell/common/MetaProposalListTest.scala index c35bafcc..fd705510 100644 --- a/modules/common/src/test/scala/docspell/common/MetaProposalListTest.scala +++ b/modules/common/src/test/scala/docspell/common/MetaProposalListTest.scala @@ -1,10 +1,10 @@ package docspell.common -import minitest._ +import munit._ import cats.data.NonEmptyList import docspell.common.MetaProposal.Candidate -object MetaProposalListTest extends SimpleTestSuite { +class MetaProposalListTest extends FunSuite { test("flatten retains order of candidates") { val cand1 = Candidate(IdRef(Ident.unsafe("123"), "name"), Set.empty) diff --git a/modules/common/src/test/scala/docspell/common/NerLabelSpanTest.scala b/modules/common/src/test/scala/docspell/common/NerLabelSpanTest.scala index add1376a..2d3568cf 100644 --- a/modules/common/src/test/scala/docspell/common/NerLabelSpanTest.scala +++ b/modules/common/src/test/scala/docspell/common/NerLabelSpanTest.scala @@ -1,8 +1,8 @@ package docspell.common -import minitest._ +import munit._ -object NerLabelSpanTest extends SimpleTestSuite { +class NerLabelSpanTest extends FunSuite { test("build") { val labels = List( diff --git a/modules/convert/src/test/scala/docspell/convert/ConversionTest.scala b/modules/convert/src/test/scala/docspell/convert/ConversionTest.scala index 4d7e80ed..f0ce80a4 100644 --- a/modules/convert/src/test/scala/docspell/convert/ConversionTest.scala +++ b/modules/convert/src/test/scala/docspell/convert/ConversionTest.scala @@ -11,10 +11,10 @@ import docspell.convert.ConversionResult.Handler import docspell.convert.extern.{TesseractConfig, UnoconvConfig, WkHtmlPdfConfig} import docspell.convert.flexmark.MarkdownConfig import docspell.files.{ExampleFiles, TestFiles} -import minitest.SimpleTestSuite +import munit._ import docspell.convert.extern.OcrMyPdfConfig -object ConversionTest extends SimpleTestSuite with FileChecks { +class ConversionTest extends FunSuite with FileChecks { val blocker = TestFiles.blocker implicit val CS = TestFiles.CS @@ -101,59 +101,58 @@ object ConversionTest extends SimpleTestSuite with FileChecks { ) test("convert to pdf") { - if (!commandsExist) ignore("At least one of the conversion programs not found") - else - File - .withTempDir[IO](target, "convpdf") - .use { dir => - conversion.use { conv => - def check(n: Long): Handler[IO, Unit] = - storePdfHandler(dir.resolve(s"test-$n.pdf")).map { p => - assert(p.isNonEmpty && p.isPDF) - } + assume(commandsExist, "At least one of the conversion programs not found") - runConversion(pdfOnly, check, conv).compile.drain - } + File + .withTempDir[IO](target, "convpdf") + .use { dir => + conversion.use { conv => + def check(n: Long): Handler[IO, Unit] = + storePdfHandler(dir.resolve(s"test-$n.pdf")).map { p => + assert(p.isNonEmpty && p.isPDF) + } + + runConversion(pdfOnly, check, conv).compile.drain } - .unsafeRunSync() + } + .unsafeRunSync() } test("convert image to pdf and txt") { - if (!commandsExist) ignore("At least one of the conversion programs not found") - else - File - .withTempDir[IO](target, "convimgpdf") - .use { dir => - conversion.use { conv => - def check(n: Long): Handler[IO, Unit] = - storePdfTxtHandler(dir.resolve(s"test-$n.pdf"), dir.resolve(s"test-$n.txt")) - .map { case (p, t) => - assert(p.isNonEmpty && p.isPDF) - assert(t.isNonEmpty && t.isPlainText) - } + assume(commandsExist, "At least one of the conversion programs not found") + File + .withTempDir[IO](target, "convimgpdf") + .use { dir => + conversion.use { conv => + def check(n: Long): Handler[IO, Unit] = + storePdfTxtHandler(dir.resolve(s"test-$n.pdf"), dir.resolve(s"test-$n.txt")) + .map { case (p, t) => + assert(p.isNonEmpty && p.isPDF) + assert(t.isNonEmpty && t.isPlainText) + } - runConversion(pdfAndTxt, check, conv).compile.drain - } + runConversion(pdfAndTxt, check, conv).compile.drain } - .unsafeRunSync() + } + .unsafeRunSync() } test("do not convert image bombs") { - if (!commandsExist) ignore("At least one of the conversion programs not found") - else - conversion - .use { conv => - def check: Handler[IO, Unit] = - Kleisli({ - case ConversionResult.InputMalformed(_, _) => - ().pure[IO] - case cr => - IO.raiseError(new Exception(s"Unexpected result: $cr")) - }) + assume(commandsExist, "At least one of the conversion programs not found") - runConversion(bombs, _ => check, conv).compile.drain - } - .unsafeRunSync() + conversion + .use { conv => + def check: Handler[IO, Unit] = + Kleisli({ + case ConversionResult.InputMalformed(_, _) => + ().pure[IO] + case cr => + IO.raiseError(new Exception(s"Unexpected result: $cr")) + }) + + runConversion(bombs, _ => check, conv).compile.drain + } + .unsafeRunSync() } def runConversion[A]( diff --git a/modules/convert/src/test/scala/docspell/convert/extern/ExternConvTest.scala b/modules/convert/src/test/scala/docspell/convert/extern/ExternConvTest.scala index d0f6c8b7..87cb3d33 100644 --- a/modules/convert/src/test/scala/docspell/convert/extern/ExternConvTest.scala +++ b/modules/convert/src/test/scala/docspell/convert/extern/ExternConvTest.scala @@ -6,10 +6,10 @@ import cats.effect._ import docspell.common._ import docspell.convert._ import docspell.files.{ExampleFiles, TestFiles} -import minitest.SimpleTestSuite +import munit._ import java.nio.charset.StandardCharsets -object ExternConvTest extends SimpleTestSuite with FileChecks { +class ExternConvTest extends FunSuite with FileChecks { val blocker = TestFiles.blocker implicit val CS = TestFiles.CS val utf8 = StandardCharsets.UTF_8 @@ -22,26 +22,24 @@ object ExternConvTest extends SimpleTestSuite with FileChecks { Seq("-s", "A4", "--encoding", "UTF-8", "-", "{{outfile}}"), Duration.seconds(20) ) + assume(commandExists(cfg.program), s"Command ${cfg.program} not found. Ignore tests.") + File + .withTempDir[IO](target, "wkhtmltopdf") + .use(dir => + IO { + val wkCfg = WkHtmlPdfConfig(cfg, target) + val p = + WkHtmlPdf + .toPDF[IO, Path](wkCfg, 8192, utf8, SanitizeHtml.none, blocker, logger)( + ExampleFiles.letter_de_html.readURL[IO](8192, blocker), + storePdfHandler(dir.resolve("test.pdf")) + ) + .unsafeRunSync() - if (!commandExists(cfg.program)) ignore(s"Command ${cfg.program} not found") - else - File - .withTempDir[IO](target, "wkhtmltopdf") - .use(dir => - IO { - val wkCfg = WkHtmlPdfConfig(cfg, target) - val p = - WkHtmlPdf - .toPDF[IO, Path](wkCfg, 8192, utf8, SanitizeHtml.none, blocker, logger)( - ExampleFiles.letter_de_html.readURL[IO](8192, blocker), - storePdfHandler(dir.resolve("test.pdf")) - ) - .unsafeRunSync() - - assert(p.isNonEmpty && p.isPDF) - } - ) - .unsafeRunSync() + assert(p.isNonEmpty && p.isPDF) + } + ) + .unsafeRunSync() } test("convert office to pdf") { @@ -51,25 +49,24 @@ object ExternConvTest extends SimpleTestSuite with FileChecks { Duration.seconds(20) ) - if (!commandExists(cfg.program)) ignore(s"Command ${cfg.program} not found") - else - File - .withTempDir[IO](target, "unoconv") - .use(dir => - IO { - val ucCfg = UnoconvConfig(cfg, target) - val p = - Unoconv - .toPDF[IO, Path](ucCfg, 8192, blocker, logger)( - ExampleFiles.examples_sample_docx.readURL[IO](8192, blocker), - storePdfHandler(dir.resolve("test.pdf")) - ) - .unsafeRunSync() + assume(commandExists(cfg.program), s"Command ${cfg.program} not found. Ignore tests.") + File + .withTempDir[IO](target, "unoconv") + .use(dir => + IO { + val ucCfg = UnoconvConfig(cfg, target) + val p = + Unoconv + .toPDF[IO, Path](ucCfg, 8192, blocker, logger)( + ExampleFiles.examples_sample_docx.readURL[IO](8192, blocker), + storePdfHandler(dir.resolve("test.pdf")) + ) + .unsafeRunSync() - assert(p.isNonEmpty && p.isPDF) - } - ) - .unsafeRunSync() + assert(p.isNonEmpty && p.isPDF) + } + ) + .unsafeRunSync() } test("convert image to pdf") { @@ -78,27 +75,25 @@ object ExternConvTest extends SimpleTestSuite with FileChecks { Seq("{{infile}}", "out", "-l", "deu", "pdf", "txt"), Duration.seconds(20) ) + assume(commandExists(cfg.program), s"Command ${cfg.program} not found") + File + .withTempDir[IO](target, "tesseract") + .use(dir => + IO { + val tessCfg = TesseractConfig(cfg, target) + val (pdf, txt) = + Tesseract + .toPDF[IO, (Path, Path)](tessCfg, Language.German, 8192, blocker, logger)( + ExampleFiles.camera_letter_en_jpg.readURL[IO](8192, blocker), + storePdfTxtHandler(dir.resolve("test.pdf"), dir.resolve("test.txt")) + ) + .unsafeRunSync() - if (!commandExists(cfg.program)) ignore(s"Command ${cfg.program} not found") - else - File - .withTempDir[IO](target, "tesseract") - .use(dir => - IO { - val tessCfg = TesseractConfig(cfg, target) - val (pdf, txt) = - Tesseract - .toPDF[IO, (Path, Path)](tessCfg, Language.German, 8192, blocker, logger)( - ExampleFiles.camera_letter_en_jpg.readURL[IO](8192, blocker), - storePdfTxtHandler(dir.resolve("test.pdf"), dir.resolve("test.txt")) - ) - .unsafeRunSync() - - assert(pdf.isNonEmpty && pdf.isPDF) - assert(txt.isNonEmpty && txt.isPlainText) - } - ) - .unsafeRunSync() + assert(pdf.isNonEmpty && pdf.isPDF) + assert(txt.isNonEmpty && txt.isPlainText) + } + ) + .unsafeRunSync() } } diff --git a/modules/extract/src/test/scala/docspell/extract/ocr/TextExtractionSuite.scala b/modules/extract/src/test/scala/docspell/extract/ocr/TextExtractionSuite.scala index 4693fd6b..c880ad6c 100644 --- a/modules/extract/src/test/scala/docspell/extract/ocr/TextExtractionSuite.scala +++ b/modules/extract/src/test/scala/docspell/extract/ocr/TextExtractionSuite.scala @@ -3,15 +3,14 @@ package docspell.extract.ocr import cats.effect.IO import docspell.common.Logger import docspell.files.TestFiles -import minitest.SimpleTestSuite +import munit._ -object TextExtractionSuite extends SimpleTestSuite { +class TextExtractionSuite extends FunSuite { import TestFiles._ val logger = Logger.log4s[IO](org.log4s.getLogger) - test("extract english pdf") { - ignore() + test("extract english pdf".ignore) { val text = TextExtract .extract[IO](letterSourceEN, blocker, logger, "eng", OcrConfig.default) .compile @@ -20,8 +19,7 @@ object TextExtractionSuite extends SimpleTestSuite { println(text) } - test("extract german pdf") { - ignore() + test("extract german pdf".ignore) { val expect = TestFiles.letterDEText val extract = TextExtract .extract[IO](letterSourceDE, blocker, logger, "deu", OcrConfig.default) diff --git a/modules/extract/src/test/scala/docspell/extract/odf/OdfExtractTest.scala b/modules/extract/src/test/scala/docspell/extract/odf/OdfExtractTest.scala index 5de1fe48..ea1d8a49 100644 --- a/modules/extract/src/test/scala/docspell/extract/odf/OdfExtractTest.scala +++ b/modules/extract/src/test/scala/docspell/extract/odf/OdfExtractTest.scala @@ -2,9 +2,9 @@ package docspell.extract.odf import cats.effect._ import docspell.files.{ExampleFiles, TestFiles} -import minitest.SimpleTestSuite +import munit._ -object OdfExtractTest extends SimpleTestSuite { +class OdfExtractTest extends FunSuite { val blocker = TestFiles.blocker implicit val CS = TestFiles.CS diff --git a/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfMetaDataTest.scala b/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfMetaDataTest.scala index b3cfb12d..4ea52331 100644 --- a/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfMetaDataTest.scala +++ b/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfMetaDataTest.scala @@ -1,8 +1,8 @@ package docspell.extract.pdfbox -import minitest.SimpleTestSuite +import munit._ -object PdfMetaDataTest extends SimpleTestSuite { +class PdfMetaDataTest extends FunSuite { test("split keywords on comma") { val md = PdfMetaData.empty.copy(keywords = Some("a,b, c")) diff --git a/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfboxExtractTest.scala b/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfboxExtractTest.scala index d9115b71..d39ef1c2 100644 --- a/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfboxExtractTest.scala +++ b/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfboxExtractTest.scala @@ -2,9 +2,9 @@ package docspell.extract.pdfbox import cats.effect._ import docspell.files.{ExampleFiles, TestFiles} -import minitest.SimpleTestSuite +import munit._ -object PdfboxExtractTest extends SimpleTestSuite { +class PdfboxExtractTest extends FunSuite { val blocker = TestFiles.blocker implicit val CS = TestFiles.CS diff --git a/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfboxPreviewTest.scala b/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfboxPreviewTest.scala index 19ded5ea..b6f356b5 100644 --- a/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfboxPreviewTest.scala +++ b/modules/extract/src/test/scala/docspell/extract/pdfbox/PdfboxPreviewTest.scala @@ -2,11 +2,11 @@ package docspell.extract.pdfbox import cats.effect._ import docspell.files.{ExampleFiles, TestFiles} -import minitest.SimpleTestSuite +import munit._ import java.nio.file.Path import fs2.Stream -object PdfboxPreviewTest extends SimpleTestSuite { +class PdfboxPreviewTest extends FunSuite { val blocker = TestFiles.blocker implicit val CS = TestFiles.CS diff --git a/modules/extract/src/test/scala/docspell/extract/poi/PoiExtractTest.scala b/modules/extract/src/test/scala/docspell/extract/poi/PoiExtractTest.scala index 0af7e8fb..d89d2c84 100644 --- a/modules/extract/src/test/scala/docspell/extract/poi/PoiExtractTest.scala +++ b/modules/extract/src/test/scala/docspell/extract/poi/PoiExtractTest.scala @@ -3,9 +3,9 @@ package docspell.extract.poi import cats.effect._ import docspell.common.MimeTypeHint import docspell.files.{ExampleFiles, TestFiles} -import minitest.SimpleTestSuite +import munit._ -object PoiExtractTest extends SimpleTestSuite { +class PoiExtractTest extends FunSuite { val blocker = TestFiles.blocker implicit val CS = TestFiles.CS diff --git a/modules/extract/src/test/scala/docspell/extract/rtf/RtfExtractTest.scala b/modules/extract/src/test/scala/docspell/extract/rtf/RtfExtractTest.scala index af98fec9..6736c0cd 100644 --- a/modules/extract/src/test/scala/docspell/extract/rtf/RtfExtractTest.scala +++ b/modules/extract/src/test/scala/docspell/extract/rtf/RtfExtractTest.scala @@ -1,9 +1,9 @@ package docspell.extract.rtf import docspell.files.ExampleFiles -import minitest.SimpleTestSuite +import munit._ -object RtfExtractTest extends SimpleTestSuite { +class RtfExtractTest extends FunSuite { test("extract text from rtf using java input-stream") { val file = ExampleFiles.examples_sample_rtf diff --git a/modules/files/src/test/scala/docspell/files/ImageSizeTest.scala b/modules/files/src/test/scala/docspell/files/ImageSizeTest.scala index c717c5b0..8cbc02c9 100644 --- a/modules/files/src/test/scala/docspell/files/ImageSizeTest.scala +++ b/modules/files/src/test/scala/docspell/files/ImageSizeTest.scala @@ -2,12 +2,12 @@ package docspell.files import cats.implicits._ import cats.effect.{Blocker, IO} -import minitest.SimpleTestSuite +import munit._ import scala.concurrent.ExecutionContext import scala.util.Using -object ImageSizeTest extends SimpleTestSuite { +class ImageSizeTest extends FunSuite { val blocker = Blocker.liftExecutionContext(ExecutionContext.global) implicit val CS = IO.contextShift(ExecutionContext.global) diff --git a/modules/files/src/test/scala/docspell/files/ZipTest.scala b/modules/files/src/test/scala/docspell/files/ZipTest.scala index 05cf1866..51dbb50a 100644 --- a/modules/files/src/test/scala/docspell/files/ZipTest.scala +++ b/modules/files/src/test/scala/docspell/files/ZipTest.scala @@ -1,12 +1,12 @@ package docspell.files -import minitest._ +import munit._ import cats.effect._ import cats.implicits._ import scala.concurrent.ExecutionContext import docspell.common.Glob -object ZipTest extends SimpleTestSuite { +class ZipTest extends FunSuite { val blocker = Blocker.liftExecutionContext(ExecutionContext.global) implicit val CS = IO.contextShift(ExecutionContext.global) diff --git a/modules/joex/src/test/scala/docspell/joex/analysis/NerFileTest.scala b/modules/joex/src/test/scala/docspell/joex/analysis/NerFileTest.scala index 03cdcbf4..fa8412a4 100644 --- a/modules/joex/src/test/scala/docspell/joex/analysis/NerFileTest.scala +++ b/modules/joex/src/test/scala/docspell/joex/analysis/NerFileTest.scala @@ -1,10 +1,10 @@ package docspell.joex.analysis -import minitest._ +import munit._ import NerFile.Pattern import java.{util => ju} -object NerFileTest extends SimpleTestSuite { +class NerFileTest extends FunSuite { test("create valid case insensitive patterns") { val names = List( diff --git a/modules/joex/src/test/scala/docspell/joex/scheduler/CountingSchemeSpec.scala b/modules/joex/src/test/scala/docspell/joex/scheduler/CountingSchemeSpec.scala index 5d867e87..56a5352d 100644 --- a/modules/joex/src/test/scala/docspell/joex/scheduler/CountingSchemeSpec.scala +++ b/modules/joex/src/test/scala/docspell/joex/scheduler/CountingSchemeSpec.scala @@ -1,9 +1,9 @@ package docspell.joex.scheduler import docspell.common.Priority -import minitest.SimpleTestSuite +import munit._ -object CountingSchemeSpec extends SimpleTestSuite { +class CountingSchemeSpec extends FunSuite { test("counting") { val cs = CountingScheme(2, 1) diff --git a/modules/store/src/test/scala/docspell/store/generator/ItemQueryGeneratorTest.scala b/modules/store/src/test/scala/docspell/store/generator/ItemQueryGeneratorTest.scala index ebc8fd33..74612153 100644 --- a/modules/store/src/test/scala/docspell/store/generator/ItemQueryGeneratorTest.scala +++ b/modules/store/src/test/scala/docspell/store/generator/ItemQueryGeneratorTest.scala @@ -3,14 +3,14 @@ package docspell.store.generator import java.time.LocalDate import docspell.store.records._ -import minitest._ +import munit._ import docspell.common._ import docspell.query.ItemQueryParser import docspell.store.queries.AttachCountTable import docspell.store.qb.DSL._ import docspell.store.qb.generator.{ItemQueryGenerator, Tables} -object ItemQueryGeneratorTest extends SimpleTestSuite { +class ItemQueryGeneratorTest extends FunSuite { import docspell.store.impl.DoobieMeta._ val tables = Tables( diff --git a/modules/store/src/test/scala/docspell/store/qb/QueryBuilderTest.scala b/modules/store/src/test/scala/docspell/store/qb/QueryBuilderTest.scala index edbe8769..f7d8b990 100644 --- a/modules/store/src/test/scala/docspell/store/qb/QueryBuilderTest.scala +++ b/modules/store/src/test/scala/docspell/store/qb/QueryBuilderTest.scala @@ -1,10 +1,10 @@ package docspell.store.qb -import minitest._ +import munit._ import docspell.store.qb.model._ import docspell.store.qb.DSL._ -object QueryBuilderTest extends SimpleTestSuite { +class QueryBuilderTest extends FunSuite { test("simple") { val c = CourseRecord.as("c") diff --git a/modules/store/src/test/scala/docspell/store/qb/impl/ConditionBuilderTest.scala b/modules/store/src/test/scala/docspell/store/qb/impl/ConditionBuilderTest.scala index 1e035f71..0d8bc32f 100644 --- a/modules/store/src/test/scala/docspell/store/qb/impl/ConditionBuilderTest.scala +++ b/modules/store/src/test/scala/docspell/store/qb/impl/ConditionBuilderTest.scala @@ -1,11 +1,11 @@ package docspell.store.qb.impl -import minitest._ +import munit._ import docspell.store.qb._ import docspell.store.qb.DSL._ import docspell.store.qb.model.{CourseRecord, PersonRecord} -object ConditionBuilderTest extends SimpleTestSuite { +class ConditionBuilderTest extends FunSuite { val c = CourseRecord.as("c") val p = PersonRecord.as("p") diff --git a/modules/store/src/test/scala/docspell/store/qb/impl/DSLTest.scala b/modules/store/src/test/scala/docspell/store/qb/impl/DSLTest.scala index 4a3f062e..8a8872d1 100644 --- a/modules/store/src/test/scala/docspell/store/qb/impl/DSLTest.scala +++ b/modules/store/src/test/scala/docspell/store/qb/impl/DSLTest.scala @@ -1,11 +1,11 @@ package docspell.store.qb.impl -import minitest._ +import munit._ import docspell.store.qb._ import docspell.store.qb.DSL._ import docspell.store.qb.model.{CourseRecord, PersonRecord} -object DSLTest extends SimpleTestSuite { +class DSLTest extends FunSuite { val course = CourseRecord.as("c") val person = PersonRecord.as("p") diff --git a/modules/store/src/test/scala/docspell/store/qb/impl/SelectBuilderTest.scala b/modules/store/src/test/scala/docspell/store/qb/impl/SelectBuilderTest.scala index f3fc8a9e..976cf4de 100644 --- a/modules/store/src/test/scala/docspell/store/qb/impl/SelectBuilderTest.scala +++ b/modules/store/src/test/scala/docspell/store/qb/impl/SelectBuilderTest.scala @@ -1,11 +1,11 @@ package docspell.store.qb.impl -import minitest._ +import munit._ import docspell.store.qb._ import docspell.store.qb.model._ import docspell.store.qb.DSL._ -object SelectBuilderTest extends SimpleTestSuite { +class SelectBuilderTest extends FunSuite { test("basic fragment") { val c = CourseRecord.as("c") diff --git a/modules/store/src/test/scala/docspell/store/queries/QueryWildcardTest.scala b/modules/store/src/test/scala/docspell/store/queries/QueryWildcardTest.scala index 91961d1f..92b92834 100644 --- a/modules/store/src/test/scala/docspell/store/queries/QueryWildcardTest.scala +++ b/modules/store/src/test/scala/docspell/store/queries/QueryWildcardTest.scala @@ -1,8 +1,8 @@ package docspell.store.queries -import minitest._ +import munit._ -object QueryWildcardTest extends SimpleTestSuite { +class QueryWildcardTest extends FunSuite { test("replace prefix") { assertEquals("%name", QueryWildcard("*name")) From 70b5625c48b5f2ba9b5be9acc412ff9f2f6caaca Mon Sep 17 00:00:00 2001 From: Eike Kettner Date: Wed, 10 Mar 2021 22:14:19 +0100 Subject: [PATCH 2/6] Add skip-duplicates to the request in ds.sh --- tools/ds.sh | 34 ++++++++++++++++++++++++++-------- 1 file changed, 26 insertions(+), 8 deletions(-) diff --git a/tools/ds.sh b/tools/ds.sh index d82ed67a..61a8ae8e 100755 --- a/tools/ds.sh +++ b/tools/ds.sh @@ -34,7 +34,7 @@ if [[ ${PIPESTATUS[0]} -ne 4 ]]; then fi OPTIONS=c:hsde -LONGOPTS=config:,help,skip,delete,exists +LONGOPTS=config:,help,skip,delete,exists,allow-duplicates ! PARSED=$(getopt --options=$OPTIONS --longoptions=$LONGOPTS --name "$0" -- "$@") if [[ ${PIPESTATUS[0]} -ne 0 ]]; then @@ -46,7 +46,7 @@ fi # read getopt’s output this way to handle the quoting right: eval set -- "$PARSED" -exists=n delete=n help=n config="${XDG_CONFIG_HOME:-$HOME/.config}/docspell/ds.conf" +exists=n delete=n help=n config="${XDG_CONFIG_HOME:-$HOME/.config}/docspell/ds.conf" dupes=n while true; do case "$1" in -h|--help) @@ -65,6 +65,10 @@ while true; do exists=y shift ;; + --allow-duplicates) + dupes=y + shift + ;; --) shift break @@ -93,7 +97,16 @@ checkFile() { upload_file() { tf=$($MKTEMP_CMD) rc=0 - $CURL_CMD -# -o "$tf" --stderr "$tf" -w "%{http_code}" -XPOST -F file=@"$1" "$2" | (2>&1 1>/dev/null grep 200) + META1="" + META2="" + if [ "$dupes" = "y" ]; then + META1="-F" + META2="meta={\"multiple\": false, \"skipDuplicates\": false}" + else + META1="-F" + META2="meta={\"multiple\": false, \"skipDuplicates\": true}" + fi + $CURL_CMD -# -o "$tf" --stderr "$tf" -w "%{http_code}" -XPOST $META1 "$META2" -F file=@"$1" "$2" | (2>&1 1>/dev/null grep 200) rc=$(expr $rc + $?) cat $tf | (2>&1 1>/dev/null grep '{"success":true') rc=$(expr $rc + $?) @@ -110,12 +123,16 @@ upload_file() { } upload() { - checkFile "$2" "$1" - if [ $? -eq 0 ]; then - info "File already exists at url $2" - return 0 - else + if [ "$dupes" == "y" ]; then upload_file "$1" "$2" + else + checkFile "$2" "$1" + if [ $? -eq 0 ]; then + info "File already exists at url $2" + return 0 + else + upload_file "$1" "$2" + fi fi } @@ -129,6 +146,7 @@ showUsage() { info " -d | --delete Delete the files when successfully uploaded (value: $delete)" info " -h | --help Prints this help text. (value: $help)" info " -e | --exists Checks for the existence of a file instead of uploading (value: $exists)" + info " --allow-duplicates Do not skip existing files in docspell (value: $dupes)" info "" info "Arguments:" info " One or more files to check for existence or upload." From a4a84abae5f09cb107d8d05dd77c178dfbe01ac5 Mon Sep 17 00:00:00 2001 From: Eike Kettner Date: Wed, 10 Mar 2021 22:14:55 +0100 Subject: [PATCH 3/6] Show errors from failed register request Also include a `@` in the valid chars for "idents". This allows to use an e-mail address as username. --- .../src/main/scala/docspell/common/Ident.scala | 2 +- .../webapp/src/main/elm/Page/Register/Update.elm | 14 ++++++++++++-- modules/webapp/src/main/elm/Util/Http.elm | 12 +++++++----- 3 files changed, 20 insertions(+), 8 deletions(-) diff --git a/modules/common/src/main/scala/docspell/common/Ident.scala b/modules/common/src/main/scala/docspell/common/Ident.scala index ded29dc8..3e30d9d4 100644 --- a/modules/common/src/main/scala/docspell/common/Ident.scala +++ b/modules/common/src/main/scala/docspell/common/Ident.scala @@ -26,7 +26,7 @@ object Ident { implicit val identEq: Eq[Ident] = Eq.by(_.id) - val chars: Set[Char] = (('A' to 'Z') ++ ('a' to 'z') ++ ('0' to '9') ++ "-_.").toSet + val chars: Set[Char] = (('A' to 'Z') ++ ('a' to 'z') ++ ('0' to '9') ++ "-_.@").toSet def randomUUID[F[_]: Sync]: F[Ident] = Sync[F].delay(unsafe(UUID.randomUUID.toString)) diff --git a/modules/webapp/src/main/elm/Page/Register/Update.elm b/modules/webapp/src/main/elm/Page/Register/Update.elm index e400bf58..3f6f01bc 100644 --- a/modules/webapp/src/main/elm/Page/Register/Update.elm +++ b/modules/webapp/src/main/elm/Page/Register/Update.elm @@ -1,9 +1,11 @@ module Page.Register.Update exposing (update) import Api +import Api.Model.BasicResult exposing (BasicResult) import Data.Flags exposing (Flags) import Page exposing (Page(..)) import Page.Register.Data exposing (..) +import Util.Http update : Flags -> Msg -> Model -> ( Model, Cmd Msg ) @@ -106,8 +108,16 @@ update flags msg model = , cmd ) - SubmitResp (Err _) -> - ( model, Cmd.none ) + SubmitResp (Err err) -> + let + errMsg = + Util.Http.errorToString err + + res = + BasicResult False + (errMsg ++ " Please check the form and try again.") + in + ( { model | result = Just res }, Cmd.none ) validateForm : Model -> List String diff --git a/modules/webapp/src/main/elm/Util/Http.elm b/modules/webapp/src/main/elm/Util/Http.elm index f9a030d3..20ebf5cd 100644 --- a/modules/webapp/src/main/elm/Util/Http.elm +++ b/modules/webapp/src/main/elm/Util/Http.elm @@ -164,12 +164,14 @@ errorToString : Http.Error -> String errorToString error = let f sc = - case sc of - 404 -> - "The requested resource doesn't exist." + if sc == 404 then + "The requested resource doesn't exist." - _ -> - "There was an invalid response status: " ++ String.fromInt sc + else if sc >= 400 && sc < 500 then + "Invalid input when processing the request." + + else + "There was an invalid response status: " ++ String.fromInt sc ++ "." in errorToStringStatus error f From 1a188afbd7d9f378e6a92624c14025575cc0eded Mon Sep 17 00:00:00 2001 From: Eike Kettner Date: Wed, 10 Mar 2021 23:14:35 +0100 Subject: [PATCH 4/6] Re-add ui-version to keep browser resource cache in service-worker --- .../src/main/scala/docspell/restserver/webapp/Flags.scala | 8 +++++--- .../scala/docspell/restserver/webapp/TemplateRoutes.scala | 5 +++-- 2 files changed, 8 insertions(+), 5 deletions(-) diff --git a/modules/restserver/src/main/scala/docspell/restserver/webapp/Flags.scala b/modules/restserver/src/main/scala/docspell/restserver/webapp/Flags.scala index 3966445e..bcd041d3 100644 --- a/modules/restserver/src/main/scala/docspell/restserver/webapp/Flags.scala +++ b/modules/restserver/src/main/scala/docspell/restserver/webapp/Flags.scala @@ -18,11 +18,12 @@ case class Flags( fullTextSearchEnabled: Boolean, maxPageSize: Int, maxNoteLength: Int, - showClassificationSettings: Boolean + showClassificationSettings: Boolean, + uiVersion: Int ) object Flags { - def apply(cfg: Config): Flags = + def apply(cfg: Config, uiVersion: Int): Flags = Flags( cfg.appName, getBaseUrl(cfg), @@ -32,7 +33,8 @@ object Flags { cfg.fullTextSearch.enabled, cfg.maxItemPageSize, cfg.maxNoteLength, - cfg.showClassificationSettings + cfg.showClassificationSettings, + uiVersion ) private def getBaseUrl(cfg: Config): String = diff --git a/modules/restserver/src/main/scala/docspell/restserver/webapp/TemplateRoutes.scala b/modules/restserver/src/main/scala/docspell/restserver/webapp/TemplateRoutes.scala index f13dd4be..b42592ac 100644 --- a/modules/restserver/src/main/scala/docspell/restserver/webapp/TemplateRoutes.scala +++ b/modules/restserver/src/main/scala/docspell/restserver/webapp/TemplateRoutes.scala @@ -133,10 +133,11 @@ object TemplateRoutes { ) object IndexData { + private[this] val uiVersion = 2 def apply(cfg: Config): IndexData = IndexData( - Flags(cfg), + Flags(cfg, uiVersion), chooseUi, Seq( "/app/assets" + Webjars.clipboardjs + "/clipboard.min.js", @@ -145,7 +146,7 @@ object TemplateRoutes { ), s"/app/assets/docspell-webapp/${BuildInfo.version}/favicon", s"/app/assets/docspell-webapp/${BuildInfo.version}/docspell.js", - Flags(cfg).asJson.spaces2 + Flags(cfg, uiVersion).asJson.spaces2 ) private def chooseUi: Seq[String] = From 0229a867af2c98a1ccbb9bec911a7c4842d8de3b Mon Sep 17 00:00:00 2001 From: Eike Kettner Date: Wed, 10 Mar 2021 22:16:04 +0100 Subject: [PATCH 5/6] Add a use colum to metadata entities --- .../scala/docspell/common/EquipmentUse.scala | 46 +++++++++++++++++++ .../main/scala/docspell/common/OrgUse.scala | 46 +++++++++++++++++++ .../scala/docspell/common/PersonUse.scala | 3 ++ .../docspell/joex/process/FindProposal.scala | 23 ++++++++-- .../restserver/conv/Conversions.scala | 10 ++-- .../migration/h2/V1.21.1__equip_org_use.sql | 16 +++++++ .../mariadb/V1.21.1__equip_org_use.sql | 16 +++++++ .../postgresql/V1.21.1__equip_org_use.sql | 16 +++++++ .../docspell/store/impl/DoobieMeta.scala | 6 +++ .../docspell/store/records/REquipment.scala | 23 +++++++--- .../store/records/ROrganization.scala | 22 ++++++--- .../webapp/src/main/elm/Comp/PersonForm.elm | 3 ++ .../webapp/src/main/elm/Data/PersonUse.elm | 15 +++++- 13 files changed, 223 insertions(+), 22 deletions(-) create mode 100644 modules/common/src/main/scala/docspell/common/EquipmentUse.scala create mode 100644 modules/common/src/main/scala/docspell/common/OrgUse.scala create mode 100644 modules/store/src/main/resources/db/migration/h2/V1.21.1__equip_org_use.sql create mode 100644 modules/store/src/main/resources/db/migration/mariadb/V1.21.1__equip_org_use.sql create mode 100644 modules/store/src/main/resources/db/migration/postgresql/V1.21.1__equip_org_use.sql diff --git a/modules/common/src/main/scala/docspell/common/EquipmentUse.scala b/modules/common/src/main/scala/docspell/common/EquipmentUse.scala new file mode 100644 index 00000000..9841da59 --- /dev/null +++ b/modules/common/src/main/scala/docspell/common/EquipmentUse.scala @@ -0,0 +1,46 @@ +package docspell.common + +import cats.data.NonEmptyList + +import io.circe.Decoder +import io.circe.Encoder + +sealed trait EquipmentUse { self: Product => + + final def name: String = + self.productPrefix.toLowerCase() +} + +object EquipmentUse { + + case object Concerning extends EquipmentUse + case object Disabled extends EquipmentUse + + def concerning: EquipmentUse = Concerning + def disabled: EquipmentUse = Disabled + + val all: NonEmptyList[EquipmentUse] = + NonEmptyList.of(concerning, disabled) + + val notDisabled: NonEmptyList[EquipmentUse] = + NonEmptyList.of(concerning) + + def fromString(str: String): Either[String, EquipmentUse] = + str.toLowerCase() match { + case "concerning" => + Right(Concerning) + case "disabled" => + Right(Disabled) + case _ => + Left(s"Unknown equipment-use: $str") + } + + def unsafeFromString(str: String): EquipmentUse = + fromString(str).fold(sys.error, identity) + + implicit val jsonDecoder: Decoder[EquipmentUse] = + Decoder.decodeString.emap(fromString) + + implicit val jsonEncoder: Encoder[EquipmentUse] = + Encoder.encodeString.contramap(_.name) +} diff --git a/modules/common/src/main/scala/docspell/common/OrgUse.scala b/modules/common/src/main/scala/docspell/common/OrgUse.scala new file mode 100644 index 00000000..fc2f803e --- /dev/null +++ b/modules/common/src/main/scala/docspell/common/OrgUse.scala @@ -0,0 +1,46 @@ +package docspell.common + +import cats.data.NonEmptyList + +import io.circe.Decoder +import io.circe.Encoder + +sealed trait OrgUse { self: Product => + + final def name: String = + self.productPrefix.toLowerCase() +} + +object OrgUse { + + case object Correspondent extends OrgUse + case object Disabled extends OrgUse + + def correspondent: OrgUse = Correspondent + def disabled: OrgUse = Disabled + + val all: NonEmptyList[OrgUse] = + NonEmptyList.of(correspondent, disabled) + + val notDisabled: NonEmptyList[OrgUse] = + NonEmptyList.of(correspondent) + + def fromString(str: String): Either[String, OrgUse] = + str.toLowerCase() match { + case "correspondent" => + Right(Correspondent) + case "disabled" => + Right(Disabled) + case _ => + Left(s"Unknown organization-use: $str") + } + + def unsafeFromString(str: String): OrgUse = + fromString(str).fold(sys.error, identity) + + implicit val jsonDecoder: Decoder[OrgUse] = + Decoder.decodeString.emap(fromString) + + implicit val jsonEncoder: Encoder[OrgUse] = + Encoder.encodeString.contramap(_.name) +} diff --git a/modules/common/src/main/scala/docspell/common/PersonUse.scala b/modules/common/src/main/scala/docspell/common/PersonUse.scala index 71cd3365..d070425e 100644 --- a/modules/common/src/main/scala/docspell/common/PersonUse.scala +++ b/modules/common/src/main/scala/docspell/common/PersonUse.scala @@ -16,6 +16,7 @@ object PersonUse { case object Correspondent extends PersonUse case object Concerning extends PersonUse case object Both extends PersonUse + case object Disabled extends PersonUse def concerning: PersonUse = Concerning def correspondent: PersonUse = Correspondent @@ -35,6 +36,8 @@ object PersonUse { Right(Concerning) case "both" => Right(Both) + case "disabled" => + Right(Disabled) case _ => Left(s"Unknown person-use: $str") } diff --git a/modules/joex/src/main/scala/docspell/joex/process/FindProposal.scala b/modules/joex/src/main/scala/docspell/joex/process/FindProposal.scala index 42bbb536..a159ad34 100644 --- a/modules/joex/src/main/scala/docspell/joex/process/FindProposal.scala +++ b/modules/joex/src/main/scala/docspell/joex/process/FindProposal.scala @@ -47,7 +47,7 @@ object FindProposal { ctx.store .transact( ROrganization - .findLike(coll, mp.values.head.ref.name.toLowerCase) + .findLike(coll, mp.values.head.ref.name.toLowerCase, OrgUse.notDisabled) .map(_.headOption) ) .flatTap(oref => @@ -85,7 +85,11 @@ object FindProposal { ctx.store .transact( REquipment - .findLike(coll, mp.values.head.ref.name.toLowerCase) + .findLike( + coll, + mp.values.head.ref.name.toLowerCase, + EquipmentUse.notDisabled + ) .map(_.headOption) ) .flatTap(oref => @@ -234,7 +238,10 @@ object FindProposal { case NerTag.Organization => ctx.logger.debug(s"Looking for organizations: $value") *> ctx.store - .transact(ROrganization.findLike(ctx.args.meta.collective, value)) + .transact( + ROrganization + .findLike(ctx.args.meta.collective, value, OrgUse.notDisabled) + ) .map(MetaProposalList.from(MetaProposalType.CorrOrg, nt)) case NerTag.Person => @@ -252,7 +259,10 @@ object FindProposal { .map(MetaProposalList.from(MetaProposalType.CorrPerson, nt)) val s3 = ctx.store - .transact(ROrganization.findLike(ctx.args.meta.collective, value)) + .transact( + ROrganization + .findLike(ctx.args.meta.collective, value, OrgUse.notDisabled) + ) .map(MetaProposalList.from(MetaProposalType.CorrOrg, nt)) ctx.logger.debug(s"Looking for persons and organizations: $value") *> (for { ml0 <- s1 @@ -268,7 +278,10 @@ object FindProposal { case NerTag.Misc => ctx.logger.debug(s"Looking for equipments: $value") *> ctx.store - .transact(REquipment.findLike(ctx.args.meta.collective, value)) + .transact( + REquipment + .findLike(ctx.args.meta.collective, value, EquipmentUse.notDisabled) + ) .map(MetaProposalList.from(MetaProposalType.ConcEquip, nt)) case NerTag.Email => diff --git a/modules/restserver/src/main/scala/docspell/restserver/conv/Conversions.scala b/modules/restserver/src/main/scala/docspell/restserver/conv/Conversions.scala index cede3845..55531fff 100644 --- a/modules/restserver/src/main/scala/docspell/restserver/conv/Conversions.scala +++ b/modules/restserver/src/main/scala/docspell/restserver/conv/Conversions.scala @@ -414,7 +414,8 @@ trait Conversions { v.notes, now, now, - v.shortName.map(_.trim) + v.shortName.map(_.trim), + OrgUse.Correspondent ) } yield OOrganization.OrgAndContacts(org, cont) } @@ -439,7 +440,8 @@ trait Conversions { v.notes, v.created, now, - v.shortName.map(_.trim) + v.shortName.map(_.trim), + OrgUse.Correspondent ) } yield OOrganization.OrgAndContacts(org, cont) } @@ -632,13 +634,13 @@ trait Conversions { def newEquipment[F[_]: Sync](e: Equipment, cid: Ident): F[REquipment] = timeId.map({ case (id, now) => - REquipment(id, cid, e.name.trim, now, now, e.notes) + REquipment(id, cid, e.name.trim, now, now, e.notes, EquipmentUse.Concerning) }) def changeEquipment[F[_]: Sync](e: Equipment, cid: Ident): F[REquipment] = Timestamp .current[F] - .map(now => REquipment(e.id, cid, e.name.trim, e.created, now, e.notes)) + .map(now => REquipment(e.id, cid, e.name.trim, e.created, now, e.notes, EquipmentUse.Concerning)) // idref diff --git a/modules/store/src/main/resources/db/migration/h2/V1.21.1__equip_org_use.sql b/modules/store/src/main/resources/db/migration/h2/V1.21.1__equip_org_use.sql new file mode 100644 index 00000000..80069ac3 --- /dev/null +++ b/modules/store/src/main/resources/db/migration/h2/V1.21.1__equip_org_use.sql @@ -0,0 +1,16 @@ +ALTER TABLE "equipment" +ADD COLUMN "equip_use" varchar(254); + +UPDATE "equipment" SET "equip_use" = 'concerning'; + +ALTER TABLE "equipment" +ALTER COLUMN "equip_use" SET NOT NULL; + + +ALTER TABLE "organization" +ADD COLUMN "org_use" varchar(254); + +UPDATE "organization" SET "org_use" = 'correspondent'; + +ALTER TABLE "organization" +ALTER COLUMN "org_use" SET NOT NULL; diff --git a/modules/store/src/main/resources/db/migration/mariadb/V1.21.1__equip_org_use.sql b/modules/store/src/main/resources/db/migration/mariadb/V1.21.1__equip_org_use.sql new file mode 100644 index 00000000..8f32937a --- /dev/null +++ b/modules/store/src/main/resources/db/migration/mariadb/V1.21.1__equip_org_use.sql @@ -0,0 +1,16 @@ +ALTER TABLE `equipment` +ADD COLUMN `equip_use` varchar(254); + +UPDATE `equipment` SET `equip_use` = 'concerning'; + +ALTER TABLE `equipment` +ALTER COLUMN `equip_use` SET NOT NULL; + + +ALTER TABLE `organization` +ADD COLUMN `org_use` varchar(254); + +UPDATE `organization` SET `org_use` = 'correspondent'; + +ALTER TABLE `organization` +ALTER COLUMN `org_use` SET NOT NULL; diff --git a/modules/store/src/main/resources/db/migration/postgresql/V1.21.1__equip_org_use.sql b/modules/store/src/main/resources/db/migration/postgresql/V1.21.1__equip_org_use.sql new file mode 100644 index 00000000..80069ac3 --- /dev/null +++ b/modules/store/src/main/resources/db/migration/postgresql/V1.21.1__equip_org_use.sql @@ -0,0 +1,16 @@ +ALTER TABLE "equipment" +ADD COLUMN "equip_use" varchar(254); + +UPDATE "equipment" SET "equip_use" = 'concerning'; + +ALTER TABLE "equipment" +ALTER COLUMN "equip_use" SET NOT NULL; + + +ALTER TABLE "organization" +ADD COLUMN "org_use" varchar(254); + +UPDATE "organization" SET "org_use" = 'correspondent'; + +ALTER TABLE "organization" +ALTER COLUMN "org_use" SET NOT NULL; diff --git a/modules/store/src/main/scala/docspell/store/impl/DoobieMeta.scala b/modules/store/src/main/scala/docspell/store/impl/DoobieMeta.scala index e38e8334..e15da7ae 100644 --- a/modules/store/src/main/scala/docspell/store/impl/DoobieMeta.scala +++ b/modules/store/src/main/scala/docspell/store/impl/DoobieMeta.scala @@ -106,6 +106,12 @@ trait DoobieMeta extends EmilDoobieMeta { implicit val metaPersonUse: Meta[PersonUse] = Meta[String].timap(PersonUse.unsafeFromString)(_.name) + + implicit val metaEquipUse: Meta[EquipmentUse] = + Meta[String].timap(EquipmentUse.unsafeFromString)(_.name) + + implicit val metaOrgUse: Meta[OrgUse] = + Meta[String].timap(OrgUse.unsafeFromString)(_.name) } object DoobieMeta extends DoobieMeta { diff --git a/modules/store/src/main/scala/docspell/store/records/REquipment.scala b/modules/store/src/main/scala/docspell/store/records/REquipment.scala index 5befc011..6cb31224 100644 --- a/modules/store/src/main/scala/docspell/store/records/REquipment.scala +++ b/modules/store/src/main/scala/docspell/store/records/REquipment.scala @@ -15,7 +15,8 @@ case class REquipment( name: String, created: Timestamp, updated: Timestamp, - notes: Option[String] + notes: Option[String], + use: EquipmentUse ) {} object REquipment { @@ -28,7 +29,8 @@ object REquipment { val created = Column[Timestamp]("created", this) val updated = Column[Timestamp]("updated", this) val notes = Column[String]("notes", this) - val all = NonEmptyList.of[Column[_]](eid, cid, name, created, updated, notes) + val use = Column[EquipmentUse]("equip_use", this) + val all = NonEmptyList.of[Column[_]](eid, cid, name, created, updated, notes, use) } val T = Table(None) @@ -41,7 +43,7 @@ object REquipment { .insert( t, t.all, - fr"${v.eid},${v.cid},${v.name},${v.created},${v.updated},${v.notes}" + fr"${v.eid},${v.cid},${v.name},${v.created},${v.updated},${v.notes},${v.use}" ) } @@ -57,7 +59,8 @@ object REquipment { t.cid.setTo(v.cid), t.name.setTo(v.name), t.updated.setTo(now), - t.notes.setTo(v.notes) + t.notes.setTo(v.notes), + t.use.setTo(v.use) ) ) } yield n @@ -90,9 +93,17 @@ object REquipment { sql.query[REquipment].to[Vector] } - def findLike(coll: Ident, equipName: String): ConnectionIO[Vector[IdRef]] = { + def findLike( + coll: Ident, + equipName: String, + use: NonEmptyList[EquipmentUse] + ): ConnectionIO[Vector[IdRef]] = { val t = Table(None) - run(select(t.eid, t.name), from(t), t.cid === coll && t.name.like(equipName)) + run( + select(t.eid, t.name), + from(t), + t.cid === coll && t.name.like(equipName) && t.use.in(use) + ) .query[IdRef] .to[Vector] } diff --git a/modules/store/src/main/scala/docspell/store/records/ROrganization.scala b/modules/store/src/main/scala/docspell/store/records/ROrganization.scala index 339c07e3..bbf4e385 100644 --- a/modules/store/src/main/scala/docspell/store/records/ROrganization.scala +++ b/modules/store/src/main/scala/docspell/store/records/ROrganization.scala @@ -22,7 +22,8 @@ case class ROrganization( notes: Option[String], created: Timestamp, updated: Timestamp, - shortName: Option[String] + shortName: Option[String], + use: OrgUse ) {} object ROrganization { @@ -43,6 +44,7 @@ object ROrganization { val created = Column[Timestamp]("created", this) val updated = Column[Timestamp]("updated", this) val shortName = Column[String]("short_name", this) + val use = Column[OrgUse]("org_use", this) val all = NonEmptyList.of[Column[_]]( oid, @@ -55,7 +57,8 @@ object ROrganization { notes, created, updated, - shortName + shortName, + use ) } @@ -67,7 +70,7 @@ object ROrganization { DML.insert( T, T.all, - fr"${v.oid},${v.cid},${v.name},${v.street},${v.zip},${v.city},${v.country},${v.notes},${v.created},${v.updated},${v.shortName}" + fr"${v.oid},${v.cid},${v.name},${v.street},${v.zip},${v.city},${v.country},${v.notes},${v.created},${v.updated},${v.shortName},${v.use}" ) def update(v: ROrganization): ConnectionIO[Int] = { @@ -84,7 +87,8 @@ object ROrganization { T.country.setTo(v.country), T.notes.setTo(v.notes), T.updated.setTo(now), - T.shortName.setTo(v.shortName) + T.shortName.setTo(v.shortName), + T.use.setTo(v.use) ) ) for { @@ -109,11 +113,17 @@ object ROrganization { sql.query[ROrganization].option } - def findLike(coll: Ident, orgName: String): ConnectionIO[Vector[IdRef]] = + def findLike( + coll: Ident, + orgName: String, + use: NonEmptyList[OrgUse] + ): ConnectionIO[Vector[IdRef]] = run( select(T.oid, T.name), from(T), - T.cid === coll && (T.name.like(orgName) || T.shortName.like(orgName)) + T.cid === coll && (T.name.like(orgName) || T.shortName.like(orgName)) && T.use.in( + use + ) ) .query[IdRef] .to[Vector] diff --git a/modules/webapp/src/main/elm/Comp/PersonForm.elm b/modules/webapp/src/main/elm/Comp/PersonForm.elm index 004157db..e9b35c79 100644 --- a/modules/webapp/src/main/elm/Comp/PersonForm.elm +++ b/modules/webapp/src/main/elm/Comp/PersonForm.elm @@ -229,6 +229,9 @@ view2 mobile settings model = Data.PersonUse.Both -> text "Use as both concerning or correspondent person" + + Data.PersonUse.Disabled -> + text "Do not use for suggestions." ] ] , div [ class "mb-4" ] diff --git a/modules/webapp/src/main/elm/Data/PersonUse.elm b/modules/webapp/src/main/elm/Data/PersonUse.elm index c00342a4..fc5d2060 100644 --- a/modules/webapp/src/main/elm/Data/PersonUse.elm +++ b/modules/webapp/src/main/elm/Data/PersonUse.elm @@ -14,6 +14,7 @@ type PersonUse = Correspondent | Concerning | Both + | Disabled fromString : String -> Maybe PersonUse @@ -28,6 +29,9 @@ fromString str = "both" -> Just Both + "disabled" -> + Just Disabled + _ -> Nothing @@ -44,6 +48,9 @@ asString pu = Both -> "both" + Disabled -> + "disabled" + label : PersonUse -> String label pu = @@ -57,10 +64,13 @@ label pu = Both -> "Both" + Disabled -> + "Disabled" + all : List PersonUse all = - [ Correspondent, Concerning, Both ] + [ Correspondent, Concerning, Both, Disabled ] spanPersonList : List Person -> { concerning : List Person, correspondent : List Person } @@ -86,5 +96,8 @@ spanPersonList input = | correspondent = p :: res.correspondent , concerning = p :: res.concerning } + + Disabled -> + res in List.foldl merge init input From 274e433d9d6ef4d334db1b381a3d249b67cc2b47 Mon Sep 17 00:00:00 2001 From: Eike Kettner Date: Wed, 10 Mar 2021 23:54:04 +0100 Subject: [PATCH 6/6] Allow to change use enum for org/equipment --- build.sbt | 6 +++ .../src/main/resources/docspell-openapi.yml | 15 ++++++ .../restserver/conv/Conversions.scala | 13 ++--- .../src/main/elm/Comp/EquipmentForm.elm | 46 ++++++++++++++++ .../src/main/elm/Comp/EquipmentTable.elm | 12 +++++ modules/webapp/src/main/elm/Comp/OrgForm.elm | 46 ++++++++++++++++ modules/webapp/src/main/elm/Comp/OrgTable.elm | 12 +++++ .../webapp/src/main/elm/Comp/SearchMenu.elm | 7 ++- .../webapp/src/main/elm/Data/EquipmentUse.elm | 52 +++++++++++++++++++ modules/webapp/src/main/elm/Data/OrgUse.elm | 50 ++++++++++++++++++ 10 files changed, 252 insertions(+), 7 deletions(-) create mode 100644 modules/webapp/src/main/elm/Data/EquipmentUse.elm create mode 100644 modules/webapp/src/main/elm/Data/OrgUse.elm diff --git a/build.sbt b/build.sbt index 290f5109..4188df7a 100644 --- a/build.sbt +++ b/build.sbt @@ -213,6 +213,12 @@ val openapiScalaSettings = Seq( case "personuse" => field => field.copy(typeDef = TypeDef("PersonUse", Imports("docspell.common.PersonUse"))) + case "orguse" => + field => + field.copy(typeDef = TypeDef("OrgUse", Imports("docspell.common.OrgUse"))) + case "equipmentuse" => + field => + field.copy(typeDef = TypeDef("EquipmentUse", Imports("docspell.common.EquipmentUse"))) })) ) diff --git a/modules/restapi/src/main/resources/docspell-openapi.yml b/modules/restapi/src/main/resources/docspell-openapi.yml index 951f2395..41494740 100644 --- a/modules/restapi/src/main/resources/docspell-openapi.yml +++ b/modules/restapi/src/main/resources/docspell-openapi.yml @@ -5064,6 +5064,7 @@ components: - id - name - created + - use properties: id: type: string @@ -5076,6 +5077,12 @@ components: format: date-time notes: type: string + use: + type: string + format: equipmentuse + enum: + - concerning + - disabled ReferenceList: description: Listing of entities with their id and a name. @@ -5119,6 +5126,7 @@ components: - concerning - correspondent - both + - disabled description: | Whether this person should be used to create suggestions for the "concerning person", "correspondent" or both. @@ -5145,6 +5153,7 @@ components: - address - contacts - created + - use properties: id: type: string @@ -5165,6 +5174,12 @@ components: format: date-time shortName: type: string + use: + type: string + format: orguse + enum: + - correspondent + - disabled OrganizationList: description: | A list of full organization values. diff --git a/modules/restserver/src/main/scala/docspell/restserver/conv/Conversions.scala b/modules/restserver/src/main/scala/docspell/restserver/conv/Conversions.scala index 55531fff..b826488c 100644 --- a/modules/restserver/src/main/scala/docspell/restserver/conv/Conversions.scala +++ b/modules/restserver/src/main/scala/docspell/restserver/conv/Conversions.scala @@ -392,7 +392,8 @@ trait Conversions { v.contacts.map(mkContact).toList, ro.notes, ro.created, - ro.shortName + ro.shortName, + ro.use ) } @@ -415,7 +416,7 @@ trait Conversions { now, now, v.shortName.map(_.trim), - OrgUse.Correspondent + v.use ) } yield OOrganization.OrgAndContacts(org, cont) } @@ -441,7 +442,7 @@ trait Conversions { v.created, now, v.shortName.map(_.trim), - OrgUse.Correspondent + v.use ) } yield OOrganization.OrgAndContacts(org, cont) } @@ -630,17 +631,17 @@ trait Conversions { // equipment def mkEquipment(re: REquipment): Equipment = - Equipment(re.eid, re.name, re.created, re.notes) + Equipment(re.eid, re.name, re.created, re.notes, re.use) def newEquipment[F[_]: Sync](e: Equipment, cid: Ident): F[REquipment] = timeId.map({ case (id, now) => - REquipment(id, cid, e.name.trim, now, now, e.notes, EquipmentUse.Concerning) + REquipment(id, cid, e.name.trim, now, now, e.notes, e.use) }) def changeEquipment[F[_]: Sync](e: Equipment, cid: Ident): F[REquipment] = Timestamp .current[F] - .map(now => REquipment(e.id, cid, e.name.trim, e.created, now, e.notes, EquipmentUse.Concerning)) + .map(now => REquipment(e.id, cid, e.name.trim, e.created, now, e.notes, e.use)) // idref diff --git a/modules/webapp/src/main/elm/Comp/EquipmentForm.elm b/modules/webapp/src/main/elm/Comp/EquipmentForm.elm index 6783212c..2968db76 100644 --- a/modules/webapp/src/main/elm/Comp/EquipmentForm.elm +++ b/modules/webapp/src/main/elm/Comp/EquipmentForm.elm @@ -10,6 +10,8 @@ module Comp.EquipmentForm exposing import Api.Model.Equipment exposing (Equipment) import Comp.Basic as B +import Comp.FixedDropdown +import Data.EquipmentUse exposing (EquipmentUse) import Data.Flags exposing (Flags) import Html exposing (..) import Html.Attributes exposing (..) @@ -22,6 +24,8 @@ type alias Model = { equipment : Equipment , name : String , notes : Maybe String + , use : EquipmentUse + , useModel : Comp.FixedDropdown.Model EquipmentUse } @@ -30,6 +34,11 @@ emptyModel = { equipment = Api.Model.Equipment.empty , name = "" , notes = Nothing + , use = Data.EquipmentUse.Concerning + , useModel = + Comp.FixedDropdown.initMap + Data.EquipmentUse.label + Data.EquipmentUse.all } @@ -44,6 +53,7 @@ getEquipment model = , name = model.name , created = model.equipment.created , notes = model.notes + , use = Data.EquipmentUse.asString model.use } @@ -51,6 +61,7 @@ type Msg = SetName String | SetEquipment Equipment | SetNotes String + | UseDropdownMsg (Comp.FixedDropdown.Msg EquipmentUse) update : Flags -> Msg -> Model -> ( Model, Cmd Msg ) @@ -61,6 +72,9 @@ update _ msg model = | equipment = t , name = t.name , notes = t.notes + , use = + Data.EquipmentUse.fromString t.use + |> Maybe.withDefault Data.EquipmentUse.Concerning } , Cmd.none ) @@ -71,6 +85,16 @@ update _ msg model = SetNotes str -> ( { model | notes = Util.Maybe.fromString str }, Cmd.none ) + UseDropdownMsg lm -> + let + ( nm, mu ) = + Comp.FixedDropdown.update lm model.useModel + + newUse = + Maybe.withDefault model.use mu + in + ( { model | useModel = nm, use = newUse }, Cmd.none ) + --- View2 @@ -102,6 +126,22 @@ view2 model = ] [] ] + , div [ class "mb-4" ] + [ label + [ class S.inputLabel + ] + [ text "Use" ] + , Html.map UseDropdownMsg + (Comp.FixedDropdown.view2 (makeUseItem model) model.useModel) + , span [ class "opacity-50 text-sm" ] + [ case model.use of + Data.EquipmentUse.Concerning -> + text "Use as concerning equipment" + + Data.EquipmentUse.Disabled -> + text "Do not use for suggestions." + ] + ] , div [ class "mb-4" ] [ h3 [ class S.header3 ] [ text "Notes" @@ -116,3 +156,9 @@ view2 model = ] ] ] + + +makeUseItem : Model -> Maybe (Comp.FixedDropdown.Item EquipmentUse) +makeUseItem model = + Just <| + Comp.FixedDropdown.Item model.use (Data.EquipmentUse.label model.use) diff --git a/modules/webapp/src/main/elm/Comp/EquipmentTable.elm b/modules/webapp/src/main/elm/Comp/EquipmentTable.elm index 3524e3f9..aabaccaa 100644 --- a/modules/webapp/src/main/elm/Comp/EquipmentTable.elm +++ b/modules/webapp/src/main/elm/Comp/EquipmentTable.elm @@ -8,6 +8,7 @@ module Comp.EquipmentTable exposing import Api.Model.Equipment exposing (Equipment) import Comp.Basic as B +import Data.EquipmentUse import Data.Flags exposing (Flags) import Html exposing (..) import Html.Attributes exposing (..) @@ -57,6 +58,9 @@ view2 model = [ thead [] [ tr [] [ th [ class "" ] [] + , th [ class "text-left pr-1 md:px-2 w-20" ] + [ text "Use" + ] , th [ class "text-left" ] [ text "Name" ] ] ] @@ -72,6 +76,14 @@ renderEquipmentLine2 model equip = , class S.tableRow ] [ B.editLinkTableCell (Select equip) + , td [ class "text-left pr-1 md:px-2" ] + [ div [ class "label inline-flex text-sm" ] + [ Data.EquipmentUse.fromString equip.use + |> Maybe.withDefault Data.EquipmentUse.Concerning + |> Data.EquipmentUse.label + |> text + ] + ] , td [ class "text-left" ] [ text equip.name ] diff --git a/modules/webapp/src/main/elm/Comp/OrgForm.elm b/modules/webapp/src/main/elm/Comp/OrgForm.elm index b0d8d848..5f2584fe 100644 --- a/modules/webapp/src/main/elm/Comp/OrgForm.elm +++ b/modules/webapp/src/main/elm/Comp/OrgForm.elm @@ -12,7 +12,9 @@ import Api.Model.Organization exposing (Organization) import Comp.AddressForm import Comp.Basic as B import Comp.ContactField +import Comp.FixedDropdown import Data.Flags exposing (Flags) +import Data.OrgUse exposing (OrgUse) import Data.UiSettings exposing (UiSettings) import Html exposing (..) import Html.Attributes exposing (..) @@ -28,6 +30,8 @@ type alias Model = , contactModel : Comp.ContactField.Model , notes : Maybe String , shortName : Maybe String + , use : OrgUse + , useModel : Comp.FixedDropdown.Model OrgUse } @@ -39,6 +43,11 @@ emptyModel = , contactModel = Comp.ContactField.emptyModel , notes = Nothing , shortName = Nothing + , use = Data.OrgUse.Correspondent + , useModel = + Comp.FixedDropdown.initMap + Data.OrgUse.label + Data.OrgUse.all } @@ -59,6 +68,7 @@ getOrg model = , contacts = Comp.ContactField.getContacts model.contactModel , notes = model.notes , shortName = model.shortName + , use = Data.OrgUse.asString model.use } @@ -69,6 +79,7 @@ type Msg | ContactMsg Comp.ContactField.Msg | SetNotes String | SetShortName String + | UseDropdownMsg (Comp.FixedDropdown.Msg OrgUse) update : Flags -> Msg -> Model -> ( Model, Cmd Msg ) @@ -87,6 +98,9 @@ update flags msg model = , name = t.name , notes = t.notes , shortName = t.shortName + , use = + Data.OrgUse.fromString t.use + |> Maybe.withDefault Data.OrgUse.Correspondent } , Cmd.batch [ c1, c2 ] ) @@ -118,11 +132,27 @@ update flags msg model = , Cmd.none ) + UseDropdownMsg lm -> + let + ( nm, mu ) = + Comp.FixedDropdown.update lm model.useModel + + newUse = + Maybe.withDefault model.use mu + in + ( { model | useModel = nm, use = newUse }, Cmd.none ) + --- View2 +makeUseItem : Model -> Maybe (Comp.FixedDropdown.Item OrgUse) +makeUseItem model = + Just <| + Comp.FixedDropdown.Item model.use (Data.OrgUse.label model.use) + + view2 : Bool -> UiSettings -> Model -> Html Msg view2 mobile settings model = div [ class "flex flex-col" ] @@ -167,6 +197,22 @@ view2 mobile settings model = ] [] ] + , div [ class "mb-4" ] + [ label + [ class S.inputLabel + ] + [ text "Use" ] + , Html.map UseDropdownMsg + (Comp.FixedDropdown.view2 (makeUseItem model) model.useModel) + , span [ class "opacity-50 text-sm" ] + [ case model.use of + Data.OrgUse.Correspondent -> + text "Use as correspondent" + + Data.OrgUse.Disabled -> + text "Do not use for suggestions." + ] + ] , div [ class "mb-4" ] [ h3 [ class S.header3 ] [ text "Address" diff --git a/modules/webapp/src/main/elm/Comp/OrgTable.elm b/modules/webapp/src/main/elm/Comp/OrgTable.elm index 46ff10ab..33d64e38 100644 --- a/modules/webapp/src/main/elm/Comp/OrgTable.elm +++ b/modules/webapp/src/main/elm/Comp/OrgTable.elm @@ -9,6 +9,7 @@ module Comp.OrgTable exposing import Api.Model.Organization exposing (Organization) import Comp.Basic as B import Data.Flags exposing (Flags) +import Data.OrgUse import Html exposing (..) import Html.Attributes exposing (..) import Styles as S @@ -58,6 +59,9 @@ view2 model = [ thead [] [ tr [] [ th [ class "" ] [] + , th [ class "text-left pr-1 md:px-2" ] + [ text "Use" + ] , th [ class "text-left" ] [ text "Name" ] , th [ class "text-left hidden md:table-cell" ] [ text "Address" ] , th [ class "text-left hidden sm:table-cell" ] [ text "Contact" ] @@ -75,6 +79,14 @@ renderOrgLine2 model org = , class S.tableRow ] [ B.editLinkTableCell (Select org) + , td [ class "text-left pr-1 md:px-2" ] + [ div [ class "label inline-flex text-sm" ] + [ Data.OrgUse.fromString org.use + |> Maybe.withDefault Data.OrgUse.Correspondent + |> Data.OrgUse.label + |> text + ] + ] , td [ class "py-4 sm:py-2 pr-2 md:pr-4" ] [ text org.name ] diff --git a/modules/webapp/src/main/elm/Comp/SearchMenu.elm b/modules/webapp/src/main/elm/Comp/SearchMenu.elm index 262a2011..0c7261a2 100644 --- a/modules/webapp/src/main/elm/Comp/SearchMenu.elm +++ b/modules/webapp/src/main/elm/Comp/SearchMenu.elm @@ -33,6 +33,7 @@ import Comp.TagSelect import Data.CustomFieldChange exposing (CustomFieldValueCollect) import Data.Direction exposing (Direction) import Data.DropdownStyle as DS +import Data.EquipmentUse import Data.Fields import Data.Flags exposing (Flags) import Data.ItemQuery as Q exposing (ItemQuery) @@ -490,7 +491,11 @@ updateDrop ddm flags settings msg model = SetConcEquip id -> let equip = - Equipment id.id id.name 0 Nothing + Equipment id.id + id.name + 0 + Nothing + (Data.EquipmentUse.asString Data.EquipmentUse.Concerning) in resetAndSet (ConcEquipmentMsg (Comp.Dropdown.SetSelection [ equip ])) diff --git a/modules/webapp/src/main/elm/Data/EquipmentUse.elm b/modules/webapp/src/main/elm/Data/EquipmentUse.elm new file mode 100644 index 00000000..7341a2fd --- /dev/null +++ b/modules/webapp/src/main/elm/Data/EquipmentUse.elm @@ -0,0 +1,52 @@ +module Data.EquipmentUse exposing + ( EquipmentUse(..) + , all + , asString + , fromString + , label + ) + +import Api.Model.Equipment exposing (Equipment) + + +type EquipmentUse + = Concerning + | Disabled + + +fromString : String -> Maybe EquipmentUse +fromString str = + case String.toLower str of + "concerning" -> + Just Concerning + + "disabled" -> + Just Disabled + + _ -> + Nothing + + +asString : EquipmentUse -> String +asString pu = + case pu of + Concerning -> + "concerning" + + Disabled -> + "disabled" + + +label : EquipmentUse -> String +label pu = + case pu of + Concerning -> + "Concerning" + + Disabled -> + "Disabled" + + +all : List EquipmentUse +all = + [ Concerning, Disabled ] diff --git a/modules/webapp/src/main/elm/Data/OrgUse.elm b/modules/webapp/src/main/elm/Data/OrgUse.elm new file mode 100644 index 00000000..ca0f6fec --- /dev/null +++ b/modules/webapp/src/main/elm/Data/OrgUse.elm @@ -0,0 +1,50 @@ +module Data.OrgUse exposing + ( OrgUse(..) + , all + , asString + , fromString + , label + ) + + +type OrgUse + = Correspondent + | Disabled + + +fromString : String -> Maybe OrgUse +fromString str = + case String.toLower str of + "correspondent" -> + Just Correspondent + + "disabled" -> + Just Disabled + + _ -> + Nothing + + +asString : OrgUse -> String +asString pu = + case pu of + Correspondent -> + "correspondent" + + Disabled -> + "disabled" + + +label : OrgUse -> String +label pu = + case pu of + Correspondent -> + "Correspondent" + + Disabled -> + "Disabled" + + +all : List OrgUse +all = + [ Correspondent, Disabled ]