mirror of
https://github.com/TheAnachronism/docspell.git
synced 2025-04-05 19:09:32 +00:00
Merge pull request #1023 from eikek/feature/983-attachments-only
Feature/983 attachments only
This commit is contained in:
commit
2cec8822c7
@ -73,7 +73,8 @@ object OUpload {
|
|||||||
skipDuplicates: Boolean,
|
skipDuplicates: Boolean,
|
||||||
fileFilter: Glob,
|
fileFilter: Glob,
|
||||||
tags: List[String],
|
tags: List[String],
|
||||||
language: Option[Language]
|
language: Option[Language],
|
||||||
|
attachmentsOnly: Option[Boolean]
|
||||||
)
|
)
|
||||||
|
|
||||||
case class UploadData[F[_]](
|
case class UploadData[F[_]](
|
||||||
@ -150,7 +151,8 @@ object OUpload {
|
|||||||
data.meta.skipDuplicates,
|
data.meta.skipDuplicates,
|
||||||
data.meta.fileFilter.some,
|
data.meta.fileFilter.some,
|
||||||
data.meta.tags.some,
|
data.meta.tags.some,
|
||||||
false
|
false,
|
||||||
|
data.meta.attachmentsOnly
|
||||||
)
|
)
|
||||||
args =
|
args =
|
||||||
if (data.multiple) files.map(f => ProcessItemArgs(meta, List(f)))
|
if (data.multiple) files.map(f => ProcessItemArgs(meta, List(f)))
|
||||||
|
@ -51,7 +51,8 @@ object ProcessItemArgs {
|
|||||||
skipDuplicate: Boolean,
|
skipDuplicate: Boolean,
|
||||||
fileFilter: Option[Glob],
|
fileFilter: Option[Glob],
|
||||||
tags: Option[List[String]],
|
tags: Option[List[String]],
|
||||||
reprocess: Boolean
|
reprocess: Boolean,
|
||||||
|
attachmentsOnly: Option[Boolean]
|
||||||
)
|
)
|
||||||
|
|
||||||
object ProcessMeta {
|
object ProcessMeta {
|
||||||
|
@ -44,7 +44,9 @@ case class ScanMailboxArgs(
|
|||||||
// the language for extraction and analysis
|
// the language for extraction and analysis
|
||||||
language: Option[Language],
|
language: Option[Language],
|
||||||
// apply additional filter to all mails or only imported
|
// apply additional filter to all mails or only imported
|
||||||
postHandleAll: Option[Boolean]
|
postHandleAll: Option[Boolean],
|
||||||
|
// Exclude the mail body when importing
|
||||||
|
attachmentsOnly: Option[Boolean]
|
||||||
)
|
)
|
||||||
|
|
||||||
object ScanMailboxArgs {
|
object ScanMailboxArgs {
|
||||||
|
@ -23,9 +23,12 @@ object ReadMail {
|
|||||||
|
|
||||||
def readBytesP[F[_]: Async](
|
def readBytesP[F[_]: Async](
|
||||||
logger: Logger[F],
|
logger: Logger[F],
|
||||||
glob: Glob
|
glob: Glob,
|
||||||
|
attachmentsOnly: Boolean
|
||||||
): Pipe[F, Byte, Binary[F]] =
|
): Pipe[F, Byte, Binary[F]] =
|
||||||
_.through(bytesToMail(logger)).flatMap(mailToEntries[F](logger, glob))
|
_.through(bytesToMail(logger)).flatMap(
|
||||||
|
mailToEntries[F](logger, glob, attachmentsOnly)
|
||||||
|
)
|
||||||
|
|
||||||
def bytesToMail[F[_]: Sync](logger: Logger[F]): Pipe[F, Byte, Mail[F]] =
|
def bytesToMail[F[_]: Sync](logger: Logger[F]): Pipe[F, Byte, Mail[F]] =
|
||||||
s =>
|
s =>
|
||||||
@ -34,10 +37,30 @@ object ReadMail {
|
|||||||
|
|
||||||
def mailToEntries[F[_]: Async](
|
def mailToEntries[F[_]: Async](
|
||||||
logger: Logger[F],
|
logger: Logger[F],
|
||||||
glob: Glob
|
glob: Glob,
|
||||||
|
attachmentsOnly: Boolean
|
||||||
|
)(mail: Mail[F]): Stream[F, Binary[F]] =
|
||||||
|
Stream.eval(
|
||||||
|
logger.debug(
|
||||||
|
s"E-mail has ${mail.attachments.size} attachments and ${bodyType(mail.body)}"
|
||||||
|
)
|
||||||
|
) >>
|
||||||
|
(makeBodyEntry(logger, glob, attachmentsOnly)(mail) ++
|
||||||
|
Stream
|
||||||
|
.eval(TnefExtract.replace(mail))
|
||||||
|
.flatMap(m => Stream.emits(m.attachments.all))
|
||||||
|
.filter(a => a.filename.exists(glob.matches(caseSensitive = false)))
|
||||||
|
.map(a =>
|
||||||
|
Binary(a.filename.getOrElse("noname"), a.mimeType.toLocal, a.content)
|
||||||
|
))
|
||||||
|
|
||||||
|
private def makeBodyEntry[F[_]: Async](
|
||||||
|
logger: Logger[F],
|
||||||
|
glob: Glob,
|
||||||
|
attachmentsOnly: Boolean
|
||||||
)(mail: Mail[F]): Stream[F, Binary[F]] = {
|
)(mail: Mail[F]): Stream[F, Binary[F]] = {
|
||||||
val bodyEntry: F[Option[Binary[F]]] =
|
val bodyEntry: F[Option[Binary[F]]] =
|
||||||
if (mail.body.isEmpty) (None: Option[Binary[F]]).pure[F]
|
if (mail.body.isEmpty || attachmentsOnly) (None: Option[Binary[F]]).pure[F]
|
||||||
else {
|
else {
|
||||||
val markdownCfg = MarkdownConfig.defaultConfig
|
val markdownCfg = MarkdownConfig.defaultConfig
|
||||||
HtmlBodyView(
|
HtmlBodyView(
|
||||||
@ -49,22 +72,14 @@ object ReadMail {
|
|||||||
).map(makeHtmlBinary[F] _).map(b => Some(b))
|
).map(makeHtmlBinary[F] _).map(b => Some(b))
|
||||||
}
|
}
|
||||||
|
|
||||||
Stream.eval(
|
for {
|
||||||
logger.debug(
|
_ <- Stream.eval(logger.debug(s"Import attachments only: $attachmentsOnly"))
|
||||||
s"E-mail has ${mail.attachments.size} attachments and ${bodyType(mail.body)}"
|
bin <-
|
||||||
)
|
|
||||||
) >>
|
|
||||||
(Stream
|
|
||||||
.eval(bodyEntry)
|
|
||||||
.flatMap(e => Stream.emits(e.toSeq))
|
|
||||||
.filter(a => glob.matches(caseSensitive = false)(a.name)) ++
|
|
||||||
Stream
|
Stream
|
||||||
.eval(TnefExtract.replace(mail))
|
.eval(bodyEntry)
|
||||||
.flatMap(m => Stream.emits(m.attachments.all))
|
.flatMap(e => Stream.emits(e.toSeq))
|
||||||
.filter(a => a.filename.exists(glob.matches(caseSensitive = false)))
|
.filter(a => glob.matches(caseSensitive = false)(a.name))
|
||||||
.map(a =>
|
} yield bin
|
||||||
Binary(a.filename.getOrElse("noname"), a.mimeType.toLocal, a.content)
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private def makeHtmlBinary[F[_]](cnt: BodyContent): Binary[F] =
|
private def makeHtmlBinary[F[_]](cnt: BodyContent): Binary[F] =
|
||||||
|
@ -161,7 +161,8 @@ object ExtractArchive {
|
|||||||
.unNoneTerminate
|
.unNoneTerminate
|
||||||
.through(ctx.store.bitpeace.fetchData2(RangeDef.all))
|
.through(ctx.store.bitpeace.fetchData2(RangeDef.all))
|
||||||
|
|
||||||
val glob = ctx.args.meta.fileFilter.getOrElse(Glob.all)
|
val glob = ctx.args.meta.fileFilter.getOrElse(Glob.all)
|
||||||
|
val attachOnly = ctx.args.meta.attachmentsOnly.getOrElse(false)
|
||||||
ctx.logger.debug(s"Filtering email attachments with '${glob.asString}'") *>
|
ctx.logger.debug(s"Filtering email attachments with '${glob.asString}'") *>
|
||||||
email
|
email
|
||||||
.through(ReadMail.bytesToMail[F](ctx.logger))
|
.through(ReadMail.bytesToMail[F](ctx.logger))
|
||||||
@ -174,7 +175,7 @@ object ExtractArchive {
|
|||||||
} yield s
|
} yield s
|
||||||
|
|
||||||
ReadMail
|
ReadMail
|
||||||
.mailToEntries(ctx.logger, glob)(mail)
|
.mailToEntries(ctx.logger, glob, attachOnly)(mail)
|
||||||
.zipWithIndex
|
.zipWithIndex
|
||||||
.flatMap(handleEntry(ctx, ra, pos, archive, mId)) ++ Stream.eval(givenMeta)
|
.flatMap(handleEntry(ctx, ra, pos, archive, mId)) ++ Stream.eval(givenMeta)
|
||||||
}
|
}
|
||||||
|
@ -114,7 +114,8 @@ object ReProcessItem {
|
|||||||
false,
|
false,
|
||||||
None,
|
None,
|
||||||
None,
|
None,
|
||||||
true
|
true,
|
||||||
|
None // attachOnly (not used when reprocessing attachments)
|
||||||
),
|
),
|
||||||
Nil
|
Nil
|
||||||
).pure[F]
|
).pure[F]
|
||||||
|
@ -300,7 +300,8 @@ object ScanMailboxTask {
|
|||||||
true,
|
true,
|
||||||
args.fileFilter.getOrElse(Glob.all),
|
args.fileFilter.getOrElse(Glob.all),
|
||||||
args.tags.getOrElse(Nil),
|
args.tags.getOrElse(Nil),
|
||||||
args.language
|
args.language,
|
||||||
|
args.attachmentsOnly
|
||||||
)
|
)
|
||||||
data = OUpload.UploadData(
|
data = OUpload.UploadData(
|
||||||
multiple = false,
|
multiple = false,
|
||||||
|
@ -4336,6 +4336,10 @@ components:
|
|||||||
format: language
|
format: language
|
||||||
postHandleAll:
|
postHandleAll:
|
||||||
type: boolean
|
type: boolean
|
||||||
|
attachmentsOnly:
|
||||||
|
type: boolean
|
||||||
|
description: |
|
||||||
|
Import only the attachments e-mails and discard the body
|
||||||
|
|
||||||
ImapSettingsList:
|
ImapSettingsList:
|
||||||
description: |
|
description: |
|
||||||
@ -5282,6 +5286,14 @@ components:
|
|||||||
description: |
|
description: |
|
||||||
The `language` of the document may be specified, otherwise
|
The `language` of the document may be specified, otherwise
|
||||||
the one from settings is used.
|
the one from settings is used.
|
||||||
|
attachmentsOnly:
|
||||||
|
type: boolean
|
||||||
|
default: false
|
||||||
|
description: |
|
||||||
|
Only applies to e-mail files. If `true` then only
|
||||||
|
attachments of the e-mail are imported and the e-mail body
|
||||||
|
is discarded. E-mails that don't have any attachments are
|
||||||
|
skipped.
|
||||||
|
|
||||||
Collective:
|
Collective:
|
||||||
description: |
|
description: |
|
||||||
|
@ -337,7 +337,8 @@ trait Conversions {
|
|||||||
m.skipDuplicates.getOrElse(false),
|
m.skipDuplicates.getOrElse(false),
|
||||||
m.fileFilter.getOrElse(Glob.all),
|
m.fileFilter.getOrElse(Glob.all),
|
||||||
m.tags.map(_.items).getOrElse(Nil),
|
m.tags.map(_.items).getOrElse(Nil),
|
||||||
m.language
|
m.language,
|
||||||
|
m.attachmentsOnly
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
@ -345,7 +346,17 @@ trait Conversions {
|
|||||||
.getOrElse(
|
.getOrElse(
|
||||||
(
|
(
|
||||||
true,
|
true,
|
||||||
UploadMeta(None, sourceName, None, validFileTypes, false, Glob.all, Nil, None)
|
UploadMeta(
|
||||||
|
None,
|
||||||
|
sourceName,
|
||||||
|
None,
|
||||||
|
validFileTypes,
|
||||||
|
false,
|
||||||
|
Glob.all,
|
||||||
|
Nil,
|
||||||
|
None,
|
||||||
|
None
|
||||||
|
)
|
||||||
)
|
)
|
||||||
.pure[F]
|
.pure[F]
|
||||||
)
|
)
|
||||||
|
@ -125,7 +125,8 @@ object ScanMailboxRoutes {
|
|||||||
settings.tags.map(_.items),
|
settings.tags.map(_.items),
|
||||||
settings.subjectFilter,
|
settings.subjectFilter,
|
||||||
settings.language,
|
settings.language,
|
||||||
settings.postHandleAll
|
settings.postHandleAll,
|
||||||
|
settings.attachmentsOnly
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
@ -159,6 +160,7 @@ object ScanMailboxRoutes {
|
|||||||
task.args.fileFilter,
|
task.args.fileFilter,
|
||||||
task.args.subjectFilter,
|
task.args.subjectFilter,
|
||||||
task.args.language,
|
task.args.language,
|
||||||
task.args.postHandleAll
|
task.args.postHandleAll,
|
||||||
|
task.args.attachmentsOnly
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -83,6 +83,7 @@ type alias Model =
|
|||||||
, language : Maybe Language
|
, language : Maybe Language
|
||||||
, postHandleAll : Bool
|
, postHandleAll : Bool
|
||||||
, summary : Maybe String
|
, summary : Maybe String
|
||||||
|
, attachmentsOnly : Bool
|
||||||
, openTabs : Set String
|
, openTabs : Set String
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -166,6 +167,7 @@ type Msg
|
|||||||
| TogglePostHandleAll
|
| TogglePostHandleAll
|
||||||
| ToggleAkkordionTab String
|
| ToggleAkkordionTab String
|
||||||
| SetSummary String
|
| SetSummary String
|
||||||
|
| ToggleAttachmentsOnly
|
||||||
|
|
||||||
|
|
||||||
initWith : Flags -> ScanMailboxSettings -> ( Model, Cmd Msg )
|
initWith : Flags -> ScanMailboxSettings -> ( Model, Cmd Msg )
|
||||||
@ -212,6 +214,7 @@ initWith flags s =
|
|||||||
Comp.FixedDropdown.init Data.Language.all
|
Comp.FixedDropdown.init Data.Language.all
|
||||||
, language = Maybe.andThen Data.Language.fromString s.language
|
, language = Maybe.andThen Data.Language.fromString s.language
|
||||||
, postHandleAll = Maybe.withDefault False s.postHandleAll
|
, postHandleAll = Maybe.withDefault False s.postHandleAll
|
||||||
|
, attachmentsOnly = Maybe.withDefault False s.attachmentsOnly
|
||||||
, summary = s.summary
|
, summary = s.summary
|
||||||
}
|
}
|
||||||
, Cmd.batch
|
, Cmd.batch
|
||||||
@ -260,6 +263,7 @@ init flags =
|
|||||||
, language = Nothing
|
, language = Nothing
|
||||||
, postHandleAll = False
|
, postHandleAll = False
|
||||||
, summary = Nothing
|
, summary = Nothing
|
||||||
|
, attachmentsOnly = False
|
||||||
, openTabs = Set.singleton (tabName TabGeneral)
|
, openTabs = Set.singleton (tabName TabGeneral)
|
||||||
}
|
}
|
||||||
, Cmd.batch
|
, Cmd.batch
|
||||||
@ -327,6 +331,7 @@ makeSettings model =
|
|||||||
, language = Maybe.map Data.Language.toIso3 model.language
|
, language = Maybe.map Data.Language.toIso3 model.language
|
||||||
, postHandleAll = Just model.postHandleAll
|
, postHandleAll = Just model.postHandleAll
|
||||||
, summary = model.summary
|
, summary = model.summary
|
||||||
|
, attachmentsOnly = Just model.attachmentsOnly
|
||||||
}
|
}
|
||||||
in
|
in
|
||||||
Result.map3 make conn schedule_ infolders
|
Result.map3 make conn schedule_ infolders
|
||||||
@ -697,6 +702,12 @@ update flags msg model =
|
|||||||
, Cmd.none
|
, Cmd.none
|
||||||
)
|
)
|
||||||
|
|
||||||
|
ToggleAttachmentsOnly ->
|
||||||
|
( { model | attachmentsOnly = not model.attachmentsOnly }
|
||||||
|
, NoAction
|
||||||
|
, Cmd.none
|
||||||
|
)
|
||||||
|
|
||||||
ToggleAkkordionTab name ->
|
ToggleAkkordionTab name ->
|
||||||
let
|
let
|
||||||
tabs =
|
tabs =
|
||||||
@ -994,6 +1005,18 @@ viewAdditionalFilter2 texts model =
|
|||||||
[ Markdown.toHtml [] texts.fileFilterInfo
|
[ Markdown.toHtml [] texts.fileFilterInfo
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
|
, div [ class "mb-4" ]
|
||||||
|
[ MB.viewItem <|
|
||||||
|
MB.Checkbox
|
||||||
|
{ id = "scanmail-attachments-only"
|
||||||
|
, value = model.attachmentsOnly
|
||||||
|
, label = texts.attachmentsOnlyLabel
|
||||||
|
, tagger = \_ -> ToggleAttachmentsOnly
|
||||||
|
}
|
||||||
|
, span [ class "opacity-50 text-sm mt-1" ]
|
||||||
|
[ Markdown.toHtml [] texts.attachmentsOnlyInfo
|
||||||
|
]
|
||||||
|
]
|
||||||
, div
|
, div
|
||||||
[ class "mb-4"
|
[ class "mb-4"
|
||||||
]
|
]
|
||||||
|
@ -70,6 +70,8 @@ type alias Texts =
|
|||||||
, connectionMissing : String
|
, connectionMissing : String
|
||||||
, noProcessingFolders : String
|
, noProcessingFolders : String
|
||||||
, invalidCalEvent : String
|
, invalidCalEvent : String
|
||||||
|
, attachmentsOnlyLabel : String
|
||||||
|
, attachmentsOnlyInfo : String
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -149,6 +151,8 @@ gb =
|
|||||||
, connectionMissing = "No E-Mail connections configured. Goto E-Mail Settings to add one."
|
, connectionMissing = "No E-Mail connections configured. Goto E-Mail Settings to add one."
|
||||||
, noProcessingFolders = "No processing folders given."
|
, noProcessingFolders = "No processing folders given."
|
||||||
, invalidCalEvent = "The calendar event is not valid."
|
, invalidCalEvent = "The calendar event is not valid."
|
||||||
|
, attachmentsOnlyLabel = "Only import e-mail attachments"
|
||||||
|
, attachmentsOnlyInfo = "Discards the e-mail body and only imports the attachments."
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -223,4 +227,6 @@ kann hier ein Wert für alle festgelegt werden. Bei 'Automatisch' wird auf den S
|
|||||||
, connectionMissing = "Keine E-Mail-Verbindung definiert. Gehe zu den E-Mail-Einstellungen und füge eine hinzu."
|
, connectionMissing = "Keine E-Mail-Verbindung definiert. Gehe zu den E-Mail-Einstellungen und füge eine hinzu."
|
||||||
, noProcessingFolders = "Keine Postfachordner ausgewählt."
|
, noProcessingFolders = "Keine Postfachordner ausgewählt."
|
||||||
, invalidCalEvent = "Das Kalenderereignis ist ungültig."
|
, invalidCalEvent = "Das Kalenderereignis ist ungültig."
|
||||||
|
, attachmentsOnlyLabel = "Nur Anhänge importieren"
|
||||||
|
, attachmentsOnlyInfo = "Verwirft den E-Mail Text und importiert nur die Anhänge."
|
||||||
}
|
}
|
||||||
|
@ -52,6 +52,7 @@ specified via a JSON structure in a part with name `meta`:
|
|||||||
, tags: Maybe StringList
|
, tags: Maybe StringList
|
||||||
, fileFilter: Maybe String
|
, fileFilter: Maybe String
|
||||||
, language: Maybe String
|
, language: Maybe String
|
||||||
|
, attachmentsOnly: Maybe Bool
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -90,6 +91,10 @@ specified via a JSON structure in a part with name `meta`:
|
|||||||
- The `language` is used for processing the document(s) contained in
|
- The `language` is used for processing the document(s) contained in
|
||||||
the request. If not specified the collective's default language is
|
the request. If not specified the collective's default language is
|
||||||
used.
|
used.
|
||||||
|
- The `attachmentsOnly` property only applies to e-mail files (usually
|
||||||
|
`*.eml`). If this is `true`, then the e-mail body is discarded and
|
||||||
|
only the attachments are imported. An e-mail without any attachments
|
||||||
|
is therefore skipped.
|
||||||
|
|
||||||
|
|
||||||
# Endpoints
|
# Endpoints
|
||||||
|
Loading…
x
Reference in New Issue
Block a user