blob: 78f27c12a786f89611927bf5bec3bdd5f6216c63 [file] [log] [blame]
/****************************************************************
* 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 *
* *
* http://www.apache.org/licenses/LICENSE-2.0 *
* *
* Unless required by applicable law or agreed to in writing, *
* software distributed under the License is distributed on an *
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY *
* KIND, either express or implied. See the License for the *
* specific language governing permissions and limitations *
* under the License. *
****************************************************************/
package org.apache.james.jmap.method
import eu.timepit.refined.auto._
import javax.inject.Inject
import org.apache.james.core.Username
import org.apache.james.jmap.core.CapabilityIdentifier.{CapabilityIdentifier, JAMES_SHARES, JMAP_CORE, JMAP_QUOTA}
import org.apache.james.jmap.core.Invocation.{Arguments, MethodName}
import org.apache.james.jmap.core.{ErrorCode, Invocation, JmapRfc8621Configuration, Properties, SessionTranslator}
import org.apache.james.jmap.json.QuotaSerializer
import org.apache.james.jmap.mail.{CountResourceType, JmapQuota, OctetsResourceType, QuotaGetRequest, QuotaIdFactory, QuotaResponseGetResult}
import org.apache.james.jmap.routes.SessionSupplier
import org.apache.james.lifecycle.api.Startable
import org.apache.james.mailbox.MailboxSession
import org.apache.james.mailbox.model.QuotaRoot
import org.apache.james.mailbox.quota.{QuotaManager, UserQuotaRootResolver}
import org.apache.james.metrics.api.MetricFactory
import org.apache.james.util.ReactorUtils
import org.reactivestreams.Publisher
import play.api.libs.json.JsObject
import reactor.core.scala.publisher.{SFlux, SMono}
class QuotaGetMethod @Inject()(val metricFactory: MetricFactory,
val configuration: JmapRfc8621Configuration,
val sessionSupplier: SessionSupplier,
val sessionTranslator: SessionTranslator,
val quotaManager: QuotaManager,
val quotaRootResolver: UserQuotaRootResolver) extends MethodRequiringAccountId[QuotaGetRequest] with Startable {
override val methodName: Invocation.MethodName = MethodName("Quota/get")
override val requiredCapabilities: Set[CapabilityIdentifier] = Set(JMAP_QUOTA, JMAP_CORE)
val jmapQuotaManagerWrapper: JmapQuotaManagerWrapper = JmapQuotaManagerWrapper(quotaManager, quotaRootResolver)
private lazy val JMAP_QUOTA_DRAFT_COMPATIBILITY: Boolean = Option(System.getProperty("james.jmap.quota.draft.compatibility"))
.exists(_.toBoolean)
override def doProcess(capabilities: Set[CapabilityIdentifier], invocation: InvocationWithContext, mailboxSession: MailboxSession, request: QuotaGetRequest): Publisher[InvocationWithContext] = {
val requestedProperties: Properties = request.properties.getOrElse(JmapQuota.allProperties(JMAP_QUOTA_DRAFT_COMPATIBILITY))
(requestedProperties -- JmapQuota.allProperties(JMAP_QUOTA_DRAFT_COMPATIBILITY) match {
case invalidProperties if invalidProperties.isEmpty() => getQuotaGetResponse(request, mailboxSession.getUser, capabilities)
.map(result => result.asResponse(accountId = request.accountId))
.map(response => Invocation(
methodName = methodName,
arguments = Arguments(QuotaSerializer.serialize(response, requestedProperties).as[JsObject]),
methodCallId = invocation.invocation.methodCallId))
case invalidProperties: Properties => SMono.just(Invocation.error(
errorCode = ErrorCode.InvalidArguments,
description = s"The following properties [${invalidProperties.format()}] do not exist.",
methodCallId = invocation.invocation.methodCallId))
})
.map(InvocationWithContext(_, invocation.processingContext))
}
override def getRequest(mailboxSession: MailboxSession, invocation: Invocation): Either[Exception, QuotaGetRequest] =
QuotaSerializer.deserializeQuotaGetRequest(invocation.arguments.value).asEitherRequest
.flatMap(request => request.validate(configuration).map(_ => request))
private def getQuotaGetResponse(request: QuotaGetRequest, username: Username, capabilities: Set[CapabilityIdentifier]): SMono[QuotaResponseGetResult] =
jmapQuotaManagerWrapper.list(username, capabilities)
.collectSeq()
.map(quotas => QuotaResponseGetResult.from(quotas, request.ids.map(_.value.map(_.id).toSet)))
}
case class JmapQuotaManagerWrapper(private val quotaManager: QuotaManager,
private val quotaRootResolver: UserQuotaRootResolver) {
def list(username: Username, capabilities: Set[CapabilityIdentifier]): SFlux[JmapQuota] =
retrieveQuotaRoot(username, capabilities)
.flatMap(quotaRoot => getJmapQuota(quotaRoot), ReactorUtils.DEFAULT_CONCURRENCY)
def retrieveQuotaRoot(username: Username, capabilities: Set[CapabilityIdentifier]): SFlux[QuotaRoot] =
SMono.just(capabilities)
.map(value => value.contains(JAMES_SHARES))
.filter(containSharedCapability => containSharedCapability)
.flatMapMany(_ => SFlux(quotaRootResolver.listAllAccessibleQuotaRoots(username)))
.switchIfEmpty(SFlux.just(quotaRootResolver.forUser(username)))
private def getJmapQuota(quotaRoot: QuotaRoot): SFlux[JmapQuota] =
SMono(quotaManager.getQuotasReactive(quotaRoot))
.flatMapMany(quotas => SMono.fromCallable(() => JmapQuota.extractUserMessageCountQuota(quotas.getMessageQuota, QuotaIdFactory.from(quotaRoot, CountResourceType), quotaRoot))
.mergeWith(SMono.fromCallable(() => JmapQuota.extractUserMessageSizeQuota(quotas.getStorageQuota, QuotaIdFactory.from(quotaRoot, OctetsResourceType), quotaRoot))))
.flatMap(SMono.justOrEmpty)
}