"""Module that provides various utility methods on the portal level."""
from Acquisition import aq_inner
from email.utils import formataddr
from email.utils import parseaddr
from logging import getLogger
from plone.api.exc import CannotGetPortalError
from plone.api.exc import InvalidParameterError
from plone.api.validation import required_parameters
from plone.app.layout.navigation.root import getNavigationRootObject
from plone.registry.interfaces import IRegistry
from Products.CMFCore.interfaces import ISiteRoot
from Products.CMFCore.utils import getToolByName
from Products.statusmessages.interfaces import IStatusMessage
from zope.component import getUtility
from zope.component import providedBy
from zope.component.hooks import getSite
from zope.globalrequest import getRequest
from zope.interface.interfaces import IInterface
import datetime as dtime
import re
logger = getLogger("plone.api.portal")
try:
from Products import PrintingMailHost
except ImportError:
PrintingMailHost = None
if not PrintingMailHost:
PRINTINGMAILHOST_ENABLED = False
elif (
PrintingMailHost.ENABLED is not None
and PrintingMailHost.ENABLED.lower() in PrintingMailHost.TRUISMS
):
PRINTINGMAILHOST_ENABLED = True
elif PrintingMailHost.ENABLED is None and PrintingMailHost.DevelopmentMode is True:
PRINTINGMAILHOST_ENABLED = True
else:
# PrintingMailHost only patches in debug mode.
# plone.api.env.debug_mode cannot be used here, because .env imports this
# file
from App.config import getConfiguration
PRINTINGMAILHOST_ENABLED = getConfiguration().debug_mode
MISSING = object()
[docs]def get():
"""Get the Plone portal object out of thin air.
Without the need to import fancy Interfaces and doing multi adapter
lookups.
:returns: Plone portal object
:rtype: Portal object
:Example: :ref:`portal-get-example`
"""
closest_site = getSite()
if closest_site is not None:
for potential_portal in closest_site.aq_chain:
if ISiteRoot in providedBy(potential_portal):
return potential_portal
raise CannotGetPortalError(
"Unable to get the portal object. More info on "
"https://docs.plone.org/develop/plone.api/docs/api/exceptions.html"
"#plone.api.exc.CannotGetPortalError",
)
[docs]@required_parameters("context")
def get_navigation_root(context=None):
"""Get the navigation root object for the context.
This traverses the path up and returns the nearest navigation root.
Useful for multi-lingual installations and sites with subsites.
:param context: [required] Context on which to get the navigation root.
:type context: context object
:returns: Navigation Root
:rtype: Portal object
:Example: :ref:`portal-get-navigation-root-example`
"""
context = aq_inner(context)
return getNavigationRootObject(context, get())
[docs]@required_parameters("recipient", "subject", "body")
def send_email(
sender=None,
recipient=None,
subject=None,
body=None,
immediate=False,
):
"""Send an email.
:param sender: Email sender, 'from' field. If not set, the portal default
will be used.
:type sender: string
:param recipient: [required] Email recipient, 'to' field.
:type recipient: string
:param subject: [required] Subject of the email.
:type subject: string
:param body: [required] Body text of the email
:type body: string or python's email object
:param immediate: Send immediate or queued at transaction commit time. When
sending immediate the mail might get sent out multiple time in case of
transaction aborts and retries.
:type body: boolean
:raises:
ValueError
:Example: :ref:`portal-send-email-example`
"""
portal = get()
if not PRINTINGMAILHOST_ENABLED:
from plone.api import content
ctrlOverview = content.get_view(
context=portal,
request=portal.REQUEST,
name="overview-controlpanel",
)
if ctrlOverview.mailhost_warning():
raise ValueError("MailHost is not configured.")
encoding = get_registry_record("plone.email_charset")
if not sender:
from_address = get_registry_record("plone.email_from_address")
from_name = get_registry_record("plone.email_from_name")
sender = formataddr((from_name, from_address))
if parseaddr(sender)[1] != from_address:
# formataddr probably got confused by special characters.
sender = from_address
# If the mail headers are not properly encoded we need to extract
# them and let MailHost manage the encoding.
if isinstance(body, str):
body = body.encode(encoding)
host = get_tool("MailHost")
host.send(
body,
recipient,
sender,
subject=subject,
charset=encoding,
immediate=immediate,
)
[docs]@required_parameters("datetime")
def get_localized_time(datetime=None, long_format=False, time_only=False):
"""Display a date/time in a user-friendly way.
It should be localized to the user's preferred language.
Note that you can specify both long_format and time_only as True
(or any other value that can be converted to a boolean True
value), but time_only then wins: the long_format value is ignored.
You can also use datetime.datetime or datetime.date instead of Plone's
DateTime. In case of datetime.datetime everything works the same, in
case of datetime.date the long_format parameter is ignored and on time_only
an empty string is returned.
:param datetime: [required] Message to show.
:type datetime: DateTime, datetime or date
:param long_format: When true, show long date format. When false
(default), show the short date format.
:type long_format: boolean
:param time_only: When true, show only the time, when false
(default), show the date.
:type time_only: boolean
:returns: Localized time
:rtype: string
:raises:
ValueError
:Example: :ref:`portal-get-localized-time-example`
"""
tool = get_tool(name="translation_service")
request = getRequest()
# isinstance won't work because of date -> datetime inheritance
if type(datetime) is dtime.date:
if time_only:
return ""
datetime = dtime.datetime(datetime.year, datetime.month, datetime.day)
long_format = False
return tool.ulocalized_time(
datetime,
long_format,
time_only,
domain="plonelocales",
request=request,
)
[docs]@required_parameters("message")
def show_message(message=None, request=None, type="info"):
"""Display a status message.
:param message: [required] Message to show.
:type message: string
:param request: [required] Request.
:type request: HTTPRequest
:param type: Message type. Possible values: 'info', 'warn', 'error'
:type type: string
:raises:
ValueError
:Example: :ref:`portal-show-message-example`
"""
if request is None:
request = getRequest()
IStatusMessage(request).add(message, type=type)
[docs]@required_parameters("name")
def get_registry_record(name=None, interface=None, default=MISSING):
"""Get a record value from ``plone.app.registry``.
:param name: [required] Name
:type name: string
:param interface: interface whose attributes are plone.app.registry
settings
:type interface: zope.interface.Interface
:param default: The value returned if the record is not found
:type default: anything
:returns: Registry record value
:rtype: plone.app.registry registry record
:Example: :ref:`portal-get-registry-record-example`
"""
if not isinstance(name, str):
raise InvalidParameterError("The 'name' parameter has to be a string")
if interface is not None and not IInterface.providedBy(interface):
raise InvalidParameterError(
"The interface parameter has to derive from " "zope.interface.Interface",
)
registry = getUtility(IRegistry)
if interface is not None:
records = registry.forInterface(interface, check=False)
_marker = object()
if getattr(records, name, _marker) != _marker:
return registry[f"{interface.__identifier__}.{name}"]
if default is not MISSING:
return default
# Show all records on the interface.
records = [key for key in interface.names()]
msg = (
'Cannot find a record with name "{name}"'
" on interface {identifier}.\n"
"Did you mean?\n"
"{records}".format(
name=name,
identifier=interface.__identifier__,
records="\n".join(records),
)
)
raise InvalidParameterError(msg)
if name in registry:
return registry[name]
if default is not MISSING:
return default
# Show all records that 'look like' name.
# We don't dump the whole list, because it 1500+ items.
msg = f"Cannot find a record with name '{name}'"
records = [key for key in registry.records.keys() if name in key]
if records:
msg = (
"{message}\n"
"Did you mean?:\n"
"{records}".format(message=msg, records="\n".join(records))
)
raise InvalidParameterError(msg)
[docs]@required_parameters("name", "value")
def set_registry_record(name=None, value=None, interface=None):
"""Set a record value in the ``plone.app.registry``.
:param name: [required] Name of the record
:type name: string
:param value: [required] Value to set
:type value: python primitive
:param interface: interface whose attributes are plone.app.registry
settings
:type interface: zope.interface.Interface
:Example: :ref:`portal-set-registry-record-example`
"""
if not isinstance(name, str):
raise InvalidParameterError("The parameter 'name' has to be a string")
if interface is not None and not IInterface.providedBy(interface):
raise InvalidParameterError(
"The interface parameter has to derive from " "zope.interface.Interface",
)
registry = getUtility(IRegistry)
if interface is not None:
# confirm that the name exists on the interface
get_registry_record(name=name, interface=interface)
from zope.schema._bootstrapinterfaces import WrongType
try:
registry[interface.__identifier__ + "." + name] = value
except WrongType:
field_type = None
for field in interface.namesAndDescriptions():
if field[0] == name:
field_type = field[1]
break
raise InvalidParameterError(
"The value parameter for the field {name} needs to be "
"{of_class} instead of {of_type}".format(
name=name,
of_class=str(field_type.__class__),
of_type=type(value),
),
)
elif isinstance(name, str):
# confirm that the record exists before setting the value
get_registry_record(name)
registry[name] = value
[docs]def get_default_language():
"""Return the default language.
:returns: language identifier
:rtype: string
:Example: :ref:`portal-get-default-language-example`
"""
from plone.i18n.interfaces import ILanguageSchema
registry = getUtility(IRegistry)
settings = registry.forInterface(ILanguageSchema, prefix="plone")
return settings.default_language
[docs]def get_current_language(context=None):
"""Return the current negotiated language.
:param context: context object
:type context: object
:returns: language identifier
:rtype: string
:Example: :ref:`portal-get-current-language-example`
"""
request = getRequest()
return (
request.get("LANGUAGE", None)
or (context and aq_inner(context).Language())
or get_default_language()
)
[docs]def translate(msgid, domain="plone", lang=None):
"""Translate a message into a given language.
Default to current negotiated language if no target language specified.
:param msgid: [required] message to translate
:type msgid: string
:type msgid: zope.i18nmessageid.Message
:param domain: i18n domain to use. When ``msgid`` is an instance of ``Message``,
then the ``Message``'s domain is used.
:type domain: string
:param lang: target language
:type lang: string
:returns: translated message
:rtype: str
:Example: :ref:`portal-translate-example`
"""
translation_service = get_tool("translation_service")
if lang and re.match(r"\D{2}-\D{2}", lang):
lang = f"{lang[:2]}_{lang[-2:].upper()}"
query = {
"msgid": msgid,
"domain": domain,
"target_language": lang,
}
if lang is None:
# Pass the request, so zope.i18n.translate can negotiate the language.
query["context"] = getRequest()
return translation_service.utranslate(**query)