246 lines
7.1 KiB
Python
246 lines
7.1 KiB
Python
# loops.util
|
|
|
|
"""Utility functions.
|
|
"""
|
|
|
|
from __future__ import absolute_import
|
|
import os
|
|
from zope.publisher.browser import BrowserView
|
|
from zope import component
|
|
from zope.catalog.interfaces import ICatalog
|
|
from zope.interface import Attribute, Interface
|
|
from zope.interface import directlyProvides, directlyProvidedBy, implements
|
|
from zope.intid.interfaces import IIntIds
|
|
from zope.i18nmessageid import MessageFactory
|
|
from zope.publisher.interfaces.browser import IBrowserRequest
|
|
from zope.app.renderer.interfaces import ISource, IHTMLRenderer
|
|
from zope.app.renderer import SourceFactory
|
|
from zope.schema import vocabulary
|
|
from zope import thread
|
|
try:
|
|
import markdown
|
|
except ImportError:
|
|
markdown = None
|
|
|
|
import cybertools
|
|
from cybertools.meta.interfaces import IOptions
|
|
from loops.browser.util import html_quote
|
|
from loops.storage.compat.common import Storage
|
|
|
|
import config
|
|
from scopes.storage.db.postgres import StorageFactory
|
|
storageFactory = StorageFactory(config, storageClass=Storage)
|
|
|
|
_ = MessageFactory('loops')
|
|
|
|
renderingFactories = {
|
|
'text/plain': 'zope.source.plaintext',
|
|
'text/stx': 'zope.source.stx',
|
|
'text/structured': 'zope.source.stx',
|
|
'text/rest': 'zope.source.rest',
|
|
'text/restructured': 'zope.source.rest',
|
|
}
|
|
|
|
if markdown:
|
|
renderingFactories['text/markdown'] = 'loops.util.markdown'
|
|
|
|
class IMarkdownSource(ISource):
|
|
"""Marker interface for a restructured text source. Note that an
|
|
implementation of this interface should always derive from unicode or
|
|
behave like a unicode class."""
|
|
|
|
|
|
MarkdownSourceFactory = SourceFactory(
|
|
IMarkdownSource, _("Markdown(md))"),
|
|
_("Markdown(md) Source"))
|
|
|
|
class MarkdownToHTMLRenderer(BrowserView):
|
|
|
|
implements(IHTMLRenderer)
|
|
component.adapts(IMarkdownSource, IBrowserRequest)
|
|
|
|
def render(self, settings_overrides={}):
|
|
return markdown.markdown(self.context)
|
|
|
|
|
|
class KeywordVocabulary(vocabulary.SimpleVocabulary):
|
|
|
|
def __init__(self, items, *interfaces):
|
|
""" ``items`` may be a tuple of (token, title) or a dictionary
|
|
with corresponding elements named 'token' and 'title'.
|
|
"""
|
|
terms = []
|
|
for t in items:
|
|
if type(t) is dict:
|
|
token, title = t['token'], t['title']
|
|
else:
|
|
token, title = t
|
|
terms.append(vocabulary.SimpleTerm(token, token, title))
|
|
super(KeywordVocabulary, self).__init__(terms, *interfaces)
|
|
|
|
|
|
def removeTargetRelation(context, event):
|
|
""" Handles IRelationInvalidatedEvent by doing some cleanup work.
|
|
"""
|
|
targetIfc = context.second.proxyInterface
|
|
if targetIfc:
|
|
directlyProvides(context.first, directlyProvidedBy(context) - targetIfc)
|
|
|
|
|
|
def nl2br(text):
|
|
if not text: return text
|
|
if '\n' in text: # Unix or DOS line endings
|
|
return '<br />\n'.join(x.replace('\r', '') for x in text.split('\n'))
|
|
else: # gracefully handle Mac line endings
|
|
return '<br />\n'.join(text.split('\r'))
|
|
|
|
def toUnicode(value, encoding='UTF-8'):
|
|
if type(value) is not unicode:
|
|
try:
|
|
return value.decode(encoding)
|
|
except UnicodeDecodeError:
|
|
return value.decode('ISO8859-15')
|
|
else:
|
|
return value
|
|
|
|
|
|
# catalog and index stuff
|
|
|
|
def getCatalog(context):
|
|
from loops.common import baseObject
|
|
context = baseObject(context)
|
|
for cat in component.getAllUtilitiesRegisteredFor(ICatalog, context=context):
|
|
return cat
|
|
|
|
def reindex(obj, catalog=None):
|
|
from loops.common import baseObject
|
|
obj = baseObject(obj)
|
|
if catalog is None:
|
|
catalog = getCatalog(obj)
|
|
if catalog is not None:
|
|
catalog.index_doc(int(getUidForObject(obj)), obj)
|
|
|
|
|
|
# options => storage => records
|
|
|
|
def getStorage(context, request=None):
|
|
if request is None:
|
|
request = getRequest()
|
|
if request is None:
|
|
return makeStorage(context)
|
|
storage = request.annotations.get('storage')
|
|
if storage is None:
|
|
storage = makeStorage(context)
|
|
request.annotations['storage'] = storage
|
|
return storage
|
|
|
|
def makeStorage(context):
|
|
root = context.getLoopsRoot()
|
|
opts = IOptions(root)
|
|
schema = (opts.scopes.storage.schema or [None])[0]
|
|
return Storage(schema=schema)
|
|
|
|
def records(context, name, factory, storage=None):
|
|
root = context.getLoopsRoot()
|
|
opts = IOptions(root)
|
|
if name in (opts.scopes.storage.records or []):
|
|
if storage is None:
|
|
storage = getStorage(context)
|
|
cont = storage.create(factory)
|
|
else:
|
|
cont = root.getRecordManager().get(name)
|
|
return cont
|
|
|
|
|
|
# UID stuff
|
|
|
|
class IUid(Interface):
|
|
|
|
uid = Attribute("Unique Identifier")
|
|
|
|
|
|
def getItem(uid, intIds=None, storage=None):
|
|
if storage is not None and '-' in uid:
|
|
return storage.getItem(uid)
|
|
obj = getObjectForUid(uid, intIds=intIds)
|
|
if obj is None and storage is not None:
|
|
return getMigratedItem(uid, storage)
|
|
return obj
|
|
|
|
def getMigratedItem(uid, storage):
|
|
t = self.getUidTable
|
|
stmt = t.select().where(t.c.legacy == int(uid))
|
|
newId = storage.session.execute(stmt).first()
|
|
if newId is not None:
|
|
return storage.getItem('%s-%i' % newUid)
|
|
return None
|
|
|
|
def getObjectForUid(uid, intIds=None):
|
|
if uid == '*': # wild card
|
|
return '*'
|
|
if isinstance(uid, basestring) and not uid.isdigit(): # not a valid uid
|
|
return None
|
|
if intIds is None:
|
|
intIds = component.getUtility(IIntIds)
|
|
try:
|
|
return intIds.getObject(int(uid))
|
|
except KeyError:
|
|
return None
|
|
|
|
def getUidForObject(obj, intIds=None):
|
|
if obj == '*': # wild card
|
|
return '*'
|
|
#if hasattr(obj, 'uid'):
|
|
if IUid.providedBy(obj):
|
|
return str(obj.uid)
|
|
if intIds is None:
|
|
intIds = component.getUtility(IIntIds)
|
|
return str(intIds.queryId(obj))
|
|
|
|
def getObjectsForUids(uids, adapt=True, storage=None):
|
|
intIds = component.getUtility(IIntIds)
|
|
#result = [getObjectForUid(uid, intIds) for uid in uids]
|
|
result = [getItem(uid, intIds, storage=storage) for uid in uids]
|
|
if adapt:
|
|
from loops.common import adapted
|
|
return [adapted(obj) for obj in result if obj is not None]
|
|
return [obj for obj in result if obj is not None]
|
|
|
|
|
|
varDirectory = None
|
|
|
|
def getVarDirectory(request=None):
|
|
varDir = varDirectory
|
|
if varDir is not None:
|
|
return varDir
|
|
if request is not None:
|
|
pub = request.publication
|
|
if pub is not None:
|
|
varDir = os.path.dirname(pub.db.getName())
|
|
if varDir is None:
|
|
instanceHome = os.path.dirname(os.path.dirname(os.path.dirname(
|
|
os.path.dirname(cybertools.__file__))))
|
|
varDir = os.path.join(instanceHome, 'var')
|
|
return varDir
|
|
|
|
def getEtcDirectory(request=None):
|
|
varDir = getVarDirectory(request)
|
|
return os.path.join(os.path.dirname(varDir), 'etc')
|
|
|
|
def getLogDirectory(request=None):
|
|
varDir = getVarDirectory(request)
|
|
return os.path.join(os.path.dirname(varDir), 'log')
|
|
|
|
|
|
# store thread-local stuff
|
|
|
|
local_data = thread.local()
|
|
|
|
def saveRequest(request):
|
|
local_data.request = request
|
|
|
|
def getRequest():
|
|
try:
|
|
return local_data.request
|
|
except AttributeError:
|
|
return None
|