Source code for synapse.lib.view

import shutil
import asyncio
import logging
import collections

import synapse.exc as s_exc
import synapse.common as s_common

import synapse.lib.cell as s_cell
import synapse.lib.snap as s_snap
import synapse.lib.layer as s_layer
import synapse.lib.nexus as s_nexus
import synapse.lib.scope as s_scope
import synapse.lib.storm as s_storm
import synapse.lib.scrape as s_scrape
import synapse.lib.msgpack as s_msgpack
import synapse.lib.schemas as s_schemas
import synapse.lib.spooled as s_spooled
import synapse.lib.trigger as s_trigger
import synapse.lib.lmdbslab as s_lmdbslab
import synapse.lib.stormctrl as s_stormctrl
import synapse.lib.stormtypes as s_stormtypes

logger = logging.getLogger(__name__)

[docs] class ViewApi(s_cell.CellApi): async def __anit__(self, core, link, user, view): await s_cell.CellApi.__anit__(self, core, link, user) self.view = view layriden = view.layers[0].iden self.allowedits = user.allowed(('node',), gateiden=layriden)
[docs] async def storNodeEdits(self, edits, meta): if not self.allowedits: mesg = 'storNodeEdits() not allowed without node permission on layer.' raise s_exc.AuthDeny(mesg=mesg) if meta is None: meta = {} meta['time'] = s_common.now() meta['user'] = self.user.iden return await self.view.storNodeEdits(edits, meta)
[docs] async def syncNodeEdits2(self, offs, wait=True): await self._reqUserAllowed(('view', 'read')) # present a layer compatible API to remote callers layr = self.view.layers[0] async for item in layr.syncNodeEdits2(offs, wait=wait): yield item await asyncio.sleep(0)
[docs] @s_cell.adminapi() async def saveNodeEdits(self, edits, meta): meta['link:user'] = self.user.iden async with await self.view.snap(user=self.user) as snap: return await snap.saveNodeEdits(edits, meta)
[docs] async def getEditSize(self): await self._reqUserAllowed(('view', 'read')) return await self.view.layers[0].getEditSize()
[docs] async def getCellIden(self): return self.view.iden
[docs] class View(s_nexus.Pusher): # type: ignore ''' A view represents a cortex as seen from a specific set of layers. The view class is used to implement Copy-On-Write layers as well as interact with a subset of the layers configured in a Cortex. ''' snapctor = s_snap.Snap.anit async def __anit__(self, core, vdef): ''' Async init the view. Args: core (Cortex): The cortex that owns the view. vdef (dict): The dictionary containing the view definition. ''' self.iden = vdef.get('iden') self.bidn = s_common.uhex(self.iden) self.info = vdef self.core = core self.dirn = s_common.gendir(core.dirn, 'views', self.iden) slabpath = s_common.genpath(self.dirn, 'viewstate.lmdb') self.viewslab = await s_lmdbslab.Slab.anit(slabpath) self.viewslab.addResizeCallback(core.checkFreeSpace) self.trigqueue = self.viewslab.getSeqn('trigqueue') self.trigdict = self.core.cortexdata.getSubKeyVal(f'view:{self.iden}:trigger:') self.triggers = s_trigger.Triggers(self) for tdef in self.trigdict.values(): try: await self.triggers.load(tdef) except Exception: # pragma: no cover logger.exception(f'Failed to load trigger {tdef!r}') await s_nexus.Pusher.__anit__(self, iden=self.iden, nexsroot=core.nexsroot) self.onfini(self.viewslab.fini) self.layers = [] self.invalid = None self.parent = None # The view this view was forked from # This will be True for a view which has a merge in progress self.merging = self.info.get('merging', False) # isolate some initialization to easily override. await self._initViewLayers() self.trigtask = None await self.initTrigTask() self.mergetask = None
[docs] def reqParentQuorum(self): if self.parent is None: mesg = f'View ({self.iden}) has no parent.' raise s_exc.BadState(mesg=mesg) quorum = self.parent.info.get('quorum') if quorum is None: mesg = f'Parent view of ({self.iden}) does not require quorum voting.' raise s_exc.BadState(mesg=mesg) if self.parent.layers[0].readonly: mesg = f'Parent view of ({self.iden}) has a read-only top layer.' raise s_exc.BadState(mesg=mesg) return quorum
[docs] def reqNoParentQuorum(self): if self.parent is None: return quorum = self.parent.info.get('quorum') if quorum is not None: mesg = f'Parent view of ({self.iden}) requires quorum voting.' raise s_exc.SynErr(mesg=mesg)
async def _wipeViewMeta(self): for lkey in self.core.slab.scanKeysByPref(self.bidn, db='view:meta'): self.core.slab.delete(lkey, db='view:meta') await asyncio.sleep(0)
[docs] def getMergeRequest(self): byts = self.core.slab.get(self.bidn + b'merge:req', db='view:meta') if byts is not None: return s_msgpack.un(byts)
[docs] async def getMergingViews(self): if self.info.get('quorum') is None: mesg = f'View ({self.iden}) does not require quorum voting.' raise s_exc.BadState(mesg=mesg) idens = [] for view in list(self.core.views.values()): await asyncio.sleep(0) if view.parent == self and view.getMergeRequest() is not None: idens.append(view.iden) return idens
[docs] async def setMergeRequest(self, mergeinfo): self.reqParentQuorum() mergeinfo['iden'] = s_common.guid() mergeinfo['created'] = s_common.now() return await self._push('merge:set', mergeinfo)
[docs] def hasKids(self): for view in self.core.views.values(): if view.parent == self: return True return False
@s_nexus.Pusher.onPush('merge:set') async def _setMergeRequest(self, mergeinfo): self.reqParentQuorum() s_schemas.reqValidMerge(mergeinfo) lkey = self.bidn + b'merge:req' self.core.slab.put(lkey, s_msgpack.en(mergeinfo), db='view:meta') await self.core.feedBeholder('view:merge:request:set', {'view': self.iden, 'merge': mergeinfo}) return mergeinfo
[docs] async def setMergeComment(self, comment): return await self._push('merge:set:comment', s_common.now(), comment)
@s_nexus.Pusher.onPush('merge:set:comment') async def _setMergeRequestComment(self, updated, comment): self.reqParentQuorum() merge = self.getMergeRequest() if merge is None: mesg = 'Cannot set the comment of a merge request that does not exist.' raise s_exc.BadState(mesg=mesg) merge['updated'] = updated merge['comment'] = comment s_schemas.reqValidMerge(merge) lkey = self.bidn + b'merge:req' self.core.slab.put(lkey, s_msgpack.en(merge), db='view:meta') await self.core.feedBeholder('view:merge:set', {'view': self.iden, 'merge': merge}) return merge
[docs] async def delMergeRequest(self): return await self._push('merge:del')
@s_nexus.Pusher.onPush('merge:del') async def _delMergeRequest(self): self.reqParentQuorum() byts = self.core.slab.pop(self.bidn + b'merge:req', db='view:meta') await self._delMergeMeta() if byts is not None: merge = s_msgpack.un(byts) await self.core.feedBeholder('view:merge:request:del', {'view': self.iden, 'merge': merge}) return merge async def _delMergeMeta(self): for lkey in self.core.slab.scanKeysByPref(self.bidn + b'merge:', db='view:meta'): await asyncio.sleep(0) self.core.slab.delete(lkey, db='view:meta')
[docs] async def getMergeVotes(self): for lkey, byts in self.core.slab.scanByPref(self.bidn + b'merge:vote', db='view:meta'): await asyncio.sleep(0) yield s_msgpack.un(byts)
[docs] async def getMerges(self): ''' Yield the historical merges into this view. ''' for lkey, bidn in self.core.slab.scanByPrefBack(self.bidn + b'hist:merge:time', db='view:meta'): byts = self.core.slab.get(self.bidn + b'hist:merge:iden' + bidn, db='view:meta') if byts is not None: yield s_msgpack.un(byts) await asyncio.sleep(0)
[docs] async def tryToMerge(self, tick): # NOTE: must be called from within a nexus handler! if self.merging: # pragma: no cover return if not await self.isMergeReady(): return self.merging = True self.info['merging'] = True self.core.viewdefs.set(self.iden, self.info) layr = self.layers[0] layr.readonly = True layr.layrinfo['readonly'] = True self.core.layerdefs.set(layr.iden, layr.layrinfo) merge = self.getMergeRequest() votes = [vote async for vote in self.getMergeVotes()] merge['votes'] = votes merge['merged'] = tick tick = s_common.int64en(tick) bidn = s_common.uhex(merge.get('iden')) lkey = self.parent.bidn + b'hist:merge:iden' + bidn self.core.slab.put(lkey, s_msgpack.en(merge), db='view:meta') lkey = self.parent.bidn + b'hist:merge:time' + tick + bidn self.core.slab.put(lkey, bidn, db='view:meta') await self.core.feedBeholder('view:merge:init', {'view': self.iden, 'merge': merge, 'votes': votes}) await self.initMergeTask()
[docs] async def setMergeVote(self, vote): self.reqParentQuorum() vote['created'] = s_common.now() vote['offset'] = await self.layers[0].getEditIndx() return await self._push('merge:vote:set', vote)
[docs] def reqValidVoter(self, useriden): merge = self.getMergeRequest() if merge is None: raise s_exc.BadState(mesg=f'View ({self.iden}) does not have a merge request.') if merge.get('creator') == useriden: raise s_exc.AuthDeny(mesg='A user may not vote for their own merge request.')
@s_nexus.Pusher.onPush('merge:vote:set') async def _setMergeVote(self, vote): self.reqParentQuorum() s_schemas.reqValidVote(vote) useriden = vote.get('user') self.reqValidVoter(useriden) bidn = s_common.uhex(useriden) self.core.slab.put(self.bidn + b'merge:vote' + bidn, s_msgpack.en(vote), db='view:meta') await self.core.feedBeholder('view:merge:vote:set', {'view': self.iden, 'vote': vote}) tick = vote.get('created') await self.tryToMerge(tick) return vote
[docs] async def setMergeVoteComment(self, useriden, comment): return await self._push('merge:vote:set:comment', s_common.now(), useriden, comment)
@s_nexus.Pusher.onPush('merge:vote:set:comment') async def _setMergeVoteComment(self, tick, useriden, comment): self.reqParentQuorum() uidn = s_common.uhex(useriden) lkey = self.bidn + b'merge:vote' + uidn byts = self.core.slab.pop(lkey, db='view:meta') if byts is None: mesg = 'Cannot set the comment for a vote that does not exist.' raise s_exc.BadState(mesg=mesg) vote = s_msgpack.un(byts) vote['updated'] = tick vote['comment'] = comment self.core.slab.put(lkey, s_msgpack.en(vote), db='view:meta') await self.core.feedBeholder('view:merge:vote:set', {'view': self.iden, 'vote': vote}) return vote
[docs] async def delMergeVote(self, useriden): return await self._push('merge:vote:del', useriden, s_common.now())
@s_nexus.Pusher.onPush('merge:vote:del') async def _delMergeVote(self, useriden, tick): self.reqParentQuorum() uidn = s_common.uhex(useriden) vote = None byts = self.core.slab.pop(self.bidn + b'merge:vote' + uidn, db='view:meta') if byts is not None: vote = s_msgpack.un(byts) await self.core.feedBeholder('view:merge:vote:del', {'view': self.iden, 'vote': vote}) await self.tryToMerge(tick) return vote
[docs] async def initMergeTask(self): if not self.merging: return if not await self.core.isCellActive(): return self.mergetask = self.core.schedCoro(self.runViewMerge())
[docs] async def finiMergeTask(self): if self.mergetask is not None: self.mergetask.cancel() self.mergetask = None
[docs] async def runViewMerge(self): # run a view merge which eventually results in removing the view and top layer # this routine must be able to be resumed and may assume that the top layer is # not receiving any edits. try: # ensure there are none marked dirty await self.layers[0]._saveDirtySodes() merge = self.getMergeRequest() votes = [vote async for vote in self.getMergeVotes()] # merge edits as the merge request user meta = { 'user': merge.get('creator'), 'merge': merge.get('iden'), } async def chunked(): nodeedits = [] async for nodeedit in self.layers[0].iterLayerNodeEdits(): nodeedits.append(nodeedit) if len(nodeedits) == 10: yield nodeedits nodeedits.clear() if nodeedits: yield nodeedits total = self.layers[0].getStorNodeCount() count = 0 nextprog = 1000 await self.core.feedBeholder('view:merge:prog', {'view': self.iden, 'count': count, 'total': total, 'merge': merge, 'votes': votes}) async with await self.parent.snap(user=self.core.auth.rootuser) as snap: async for edits in chunked(): meta['time'] = s_common.now() await snap.saveNodeEdits(edits, meta) await asyncio.sleep(0) count += len(edits) if count >= nextprog: await self.core.feedBeholder('view:merge:prog', {'view': self.iden, 'count': count, 'total': total, 'merge': merge, 'votes': votes}) nextprog += 1000 await self.core.feedBeholder('view:merge:fini', {'view': self.iden, 'merge': merge, 'merge': merge, 'votes': votes}) # remove the view and top layer await self.core.delViewWithLayer(self.iden) except Exception as e: # pragma: no cover logger.exception(f'Error while merging view: {self.iden}')
[docs] async def isMergeReady(self): # count the current votes and potentially trigger a merge offset = await self.layers[0].getEditIndx() quorum = self.reqParentQuorum() approvals = 0 async for vote in self.getMergeVotes(): if vote.get('offset') != offset: continue # any disapprovals will hold merging if not vote.get('approved'): return False approvals += 1 return approvals >= quorum.get('count')
[docs] async def detach(self): ''' Detach the view from its parent but do not change the layers. ( this is not reversible! ) ''' if not self.parent: mesg = 'A view with no parent is already detached.' raise s_exc.BadArg(mesg=mesg) return await self._push('view:detach')
@s_nexus.Pusher.onPush('view:detach') async def _detach(self): # remove any pending merge requests or votes await self._delMergeMeta() self.parent = None if self.info.pop('parent', None) is not None: self.core.viewdefs.set(self.iden, self.info) await self.core.feedBeholder('view:set', {'iden': self.iden, 'name': 'parent', 'valu': None}, gates=[self.iden, self.layers[0].iden])
[docs] async def mergeStormIface(self, name, todo): ''' Allow an interface which specifies a generator use case to yield (priority, value) tuples and merge results from multiple generators yielded in ascending priority order. ''' root = self.core.auth.rootuser funcname, funcargs, funckwargs = todo genrs = [] async with await self.snap(user=root) as snap: for moddef in await self.core.getStormIfaces(name): try: query = await self.core.getStormQuery(moddef.get('storm')) modconf = moddef.get('modconf', {}) runt = await snap.addStormRuntime(query, opts={'vars': {'modconf': modconf}}, user=root) # let it initialize the function async for item in runt.execute(): await asyncio.sleep(0) func = runt.vars.get(funcname) if func is None: continue genrs.append(await func(*funcargs, **funckwargs)) except Exception as e: # pragma: no cover logger.exception('mergeStormIface()') if genrs: async for item in s_common.merggenr2(genrs): yield item
[docs] async def callStormIface(self, name, todo): root = self.core.auth.rootuser funcname, funcargs, funckwargs = todo async with await self.snap(user=root) as snap: for moddef in await self.core.getStormIfaces(name): try: query = await self.core.getStormQuery(moddef.get('storm')) modconf = moddef.get('modconf', {}) # TODO look at caching the function returned as presume a persistant runtime? async with snap.getStormRuntime(query, opts={'vars': {'modconf': modconf}}, user=root) as runt: # let it initialize the function async for item in runt.execute(): await asyncio.sleep(0) func = runt.vars.get(funcname) if func is None: continue valu = await func(*funcargs, **funckwargs) yield await s_stormtypes.toprim(valu) except Exception as e: modname = moddef.get('name') logger.exception(f'callStormIface {name} mod: {modname}')
[docs] async def initTrigTask(self): if self.trigtask is not None: return if not await self.core.isCellActive(): return self.trigtask = self.schedCoro(self._trigQueueLoop())
[docs] async def finiTrigTask(self): if self.trigtask is not None: self.trigtask.cancel() self.trigtask = None
async def _trigQueueLoop(self): while not self.isfini: async for offs, triginfo in self.trigqueue.gets(0): buid = triginfo.get('buid') varz = triginfo.get('vars') trigiden = triginfo.get('trig') try: trig = self.triggers.get(trigiden) if trig is None: continue async with await self.snap(trig.user) as snap: node = await snap.getNodeByBuid(buid) if node is None: continue await trig._execute(node, vars=varz) except asyncio.CancelledError: # pragma: no cover raise except Exception as e: # pragma: no cover logger.exception(f'trigQueueLoop() on trigger: {trigiden} view: {self.iden}') finally: await self.delTrigQueue(offs)
[docs] async def getStorNodes(self, buid): ''' Return a list of storage nodes for the given buid in layer order. ''' return await self.core._getStorNodes(buid, self.layers)
[docs] def init2(self): ''' We have a second round of initialization so the views can get a handle to their parents which might not be initialized yet ''' parent = self.info.get('parent') if parent is not None: self.parent = self.core.getView(parent)
[docs] def isafork(self): return self.parent is not None
[docs] def isForkOf(self, viewiden): view = self.parent while view is not None: if view.iden == viewiden: return True view = view.parent return False
async def _calcForkLayers(self): # recompute the proper set of layers for a forked view # (this may only be called from within a nexus handler) ''' We spent a lot of time thinking/talking about this so some hefty comments are in order: For a given stack of views (example below), only the bottom view of the stack may have more than one original layer. When adding a new view to the top of the stack (via `<viewE>.setViewInfo('parent', <viewD>)`), we grab the top layer of each of the views from View D to View B and then all of the layers from View A. We know this is the right behavior because all views above the bottom view only have one original layer (but will include all of the layers of it's parents) which is enforced by `setViewInfo()`. View D - Layer 6 (original to View D) - Layer 5 (copied from View C) - Layer 4 (copied from View B) - Layer 3 (copied from View A) - Layer 2 (copied from View A) - Layer 1 (copied from View A) View C (parent of D) - Layer 5 (original to View C) - Layer 4 (copied from View B) - Layer 3 (copied from View A) - Layer 2 (copied from View A) - Layer 1 (copied from View A) View B (parent of C) - Layer 4 (original to View B) - Layer 3 (copied from View A) - Layer 2 (copied from View A) - Layer 1 (copied from View A) View A (parent of B) - Layer 3 - Layer 2 - Layer 1 Continuing the exercise: when adding View E, it has it's own layer (Layer 7). We then copy Layer 6 from View D, Layer 5 from View C, Layer 4 from View B, and Layers 3-1 from View A (the bottom view). This gives us the new view which looks like this: View E: - Layer 7 (original to View E) - Layer 6 (copied from View D) - Layer 5 (copied from View C) - Layer 4 (copied from View B) - Layer 3 (copied from View A) - Layer 2 (copied from View A) - Layer 1 (copied from View A) View D (now parent of View E) ... (everything from View D and below is the same as above) ''' layers = [] # Add the top layer from each of the views that aren't the bottom view. # This is the view's original layer. view = self while view.parent is not None: layers.append(view.layers[0]) view = view.parent # Add all of the bottom view's layers. layers.extend(view.layers) layridens = [layr.iden for layr in layers] self.layers = layers self.info['layers'] = layridens self.core.viewdefs.set(self.iden, self.info) await self.core.feedBeholder('view:setlayers', {'iden': self.iden, 'layers': layridens}, gates=[self.iden, self.layers[0].iden])
[docs] async def pack(self): d = {'iden': self.iden} d.update(self.info) layrinfo = [await lyr.pack() for lyr in self.layers] d['layers'] = layrinfo triginfo = [t.pack() for _, t in self.triggers.list()] d['triggers'] = triginfo return d
[docs] async def getFormCounts(self): counts = collections.defaultdict(int) for layr in self.layers: for name, valu in (await layr.getFormCounts()).items(): counts[name] += valu return counts
[docs] async def getPropCount(self, propname, valu=s_common.novalu): prop = self.core.model.prop(propname) if prop is None: mesg = f'No property named {propname}' raise s_exc.NoSuchProp(mesg=mesg) count = 0 formname = None propname = None if prop.isform: formname = prop.name else: propname = prop.name if not prop.isuniv: formname = prop.form.name if valu is s_common.novalu: for layr in self.layers: await asyncio.sleep(0) count += await layr.getPropCount(formname, propname) return count norm, info = prop.type.norm(valu) for layr in self.layers: await asyncio.sleep(0) count += layr.getPropValuCount(formname, propname, prop.type.stortype, norm) return count
[docs] async def getTagPropCount(self, form, tag, propname, valu=s_common.novalu): prop = self.core.model.getTagProp(propname) if prop is None: mesg = f'No tag property named {propname}' raise s_exc.NoSuchTagProp(name=propname, mesg=mesg) count = 0 if valu is s_common.novalu: for layr in self.layers: await asyncio.sleep(0) count += await layr.getTagPropCount(form, tag, prop.name) return count norm, info = prop.type.norm(valu) for layr in self.layers: await asyncio.sleep(0) count += layr.getTagPropValuCount(form, tag, prop.name, prop.type.stortype, norm) return count
[docs] async def getPropArrayCount(self, propname, valu=s_common.novalu): prop = self.core.model.prop(propname) if prop is None: mesg = f'No property named {propname}' raise s_exc.NoSuchProp(mesg=mesg) if not prop.type.isarray: mesg = f'Property is not an array type: {prop.type.name}.' raise s_exc.BadTypeValu(mesg=mesg) count = 0 formname = None propname = None if prop.isform: formname = prop.name else: propname = prop.name if not prop.isuniv: formname = prop.form.name if valu is s_common.novalu: for layr in self.layers: await asyncio.sleep(0) count += await layr.getPropArrayCount(formname, propname) return count atyp = prop.type.arraytype norm, info = atyp.norm(valu) for layr in self.layers: await asyncio.sleep(0) count += layr.getPropArrayValuCount(formname, propname, atyp.stortype, norm) return count
[docs] async def iterPropValues(self, propname): prop = self.core.model.reqProp(propname) formname = None propname = None if prop.isform: formname = prop.name else: propname = prop.name if not prop.isuniv: formname = prop.form.name async def wrapgenr(lidx, genr): async for indx, valu in genr: yield indx, valu, lidx genrs = [] for lidx, layr in enumerate(self.layers): genr = layr.iterPropValues(formname, propname, prop.type.stortype) genrs.append(wrapgenr(lidx, genr)) lastvalu = None async for indx, valu, lidx in s_common.merggenr2(genrs): if valu == lastvalu: continue if lidx == 0 or propname is None: lastvalu = valu yield valu else: valid = False async for buid in self.layers[lidx].iterPropIndxBuids(formname, propname, indx): for layr in self.layers[0:lidx]: if (sode := layr._getStorNode(buid)) is None: continue if sode['props'].get(propname) is not None: break else: valid = True if valid: lastvalu = valu yield valu break
[docs] async def getEdgeVerbs(self): async with await s_spooled.Set.anit(dirn=self.core.dirn, cell=self.core) as vset: for layr in self.layers: async for verb in layr.getEdgeVerbs(): await asyncio.sleep(0) if verb in vset: continue await vset.add(verb) yield verb
[docs] async def getEdges(self, verb=None): async with await s_spooled.Set.anit(dirn=self.core.dirn, cell=self.core) as eset: for layr in self.layers: async for edge in layr.getEdges(verb=verb): await asyncio.sleep(0) if edge in eset: continue await eset.add(edge) yield edge
async def _initViewLayers(self): self.layers = [] for iden in self.info.get('layers'): layr = self.core.layers.get(iden) if layr is None: self.invalid = iden logger.warning('view %r has missing layer %r' % (self.iden, iden)) continue self.layers.append(layr)
[docs] async def eval(self, text, opts=None): ''' Evaluate a storm query and yield Nodes only. ''' opts = self.core._initStormOpts(opts) user = self.core._userFromOpts(opts) info = opts.get('_loginfo', {}) info.update({'mode': opts.get('mode', 'storm'), 'view': self.iden}) self.core._logStormQuery(text, user, info=info) taskiden = opts.get('task') taskinfo = {'query': text, 'view': self.iden} with s_scope.enter({'user': user}): await self.core.boss.promote('storm', user=user, info=taskinfo, taskiden=taskiden) async with await self.snap(user=user) as snap: async for node in snap.eval(text, opts=opts, user=user): yield node
[docs] async def callStorm(self, text, opts=None): user = self.core._userFromOpts(opts) try: async for item in self.eval(text, opts=opts): await asyncio.sleep(0) # pragma: no cover except s_stormctrl.StormReturn as e: # Catch return( ... ) values and return the # primitive version of that item. return await s_stormtypes.toprim(e.item) except asyncio.CancelledError: logger.warning(f'callStorm cancelled', extra={'synapse': {'text': text, 'username': user.name, 'user': user.iden}}) raise except (s_stormctrl.StormLoopCtrl, s_stormctrl.StormGenrCtrl) as e: if isinstance(e, s_stormctrl.StormLoopCtrl): mesg = f'Loop control statement "{e.statement}" used outside of a loop.' else: mesg = f'Generator control statement "{e.statement}" used outside of a generator function.' logmesg = f'Error during storm execution for {{ {text} }} - {mesg}' logger.exception(logmesg, extra={'synapse': {'text': text, 'username': user.name, 'user': user.iden}}) raise s_exc.StormRuntimeError(mesg=mesg, statement=e.statement, highlight=e.get('highlight')) from e except Exception: logger.exception(f'Error during callStorm execution for {{ {text} }}', extra={'synapse': {'text': text, 'username': user.name, 'user': user.iden}}) raise
# Any other exceptions will be raised to # callers as expected.
[docs] async def nodes(self, text, opts=None): ''' A simple non-streaming way to return a list of nodes. ''' return [n async for n in self.eval(text, opts=opts)]
[docs] async def stormlist(self, text, opts=None): # an ease-of-use API for testing return [m async for m in self.storm(text, opts=opts)]
[docs] async def storm(self, text, opts=None): ''' Evaluate a storm query and yield result messages. Yields: ((str,dict)): Storm messages. ''' if not isinstance(text, str): mesg = 'Storm query text must be a string' raise s_exc.BadArg(mesg=mesg) opts = self.core._initStormOpts(opts) user = self.core._userFromOpts(opts) MSG_QUEUE_SIZE = 1000 chan = asyncio.Queue(MSG_QUEUE_SIZE) taskinfo = {'query': text, 'view': self.iden} taskiden = opts.get('task') keepalive = opts.get('keepalive') if keepalive is not None and keepalive <= 0: raise s_exc.BadArg(mesg=f'keepalive must be > 0; got {keepalive}') synt = await self.core.boss.promote('storm', user=user, info=taskinfo, taskiden=taskiden) show = opts.get('show', set()) mode = opts.get('mode', 'storm') editformat = opts.get('editformat', 'nodeedits') if editformat not in ('nodeedits', 'count', 'none'): raise s_exc.BadConfValu(mesg=f'invalid edit format, got {editformat}', name='editformat', valu=editformat) texthash = s_storm.queryhash(text) async def runStorm(): cancelled = False tick = s_common.now() abstick = s_common.mononow() count = 0 try: # Always start with an init message. await chan.put(('init', {'tick': tick, 'text': text, 'abstick': abstick, 'hash': texthash, 'task': synt.iden})) # Try text parsing. If this fails, we won't be able to get a storm # runtime in the snap, so catch and pass the `err` message await self.core.getStormQuery(text, mode=mode) shownode = (not show or 'node' in show) with s_scope.enter({'user': user}): async with await self.snap(user=user) as snap: if keepalive: snap.schedCoro(snap.keepalive(keepalive)) if not show: snap.link(chan.put) else: [snap.on(n, chan.put) for n in show] if shownode: async for pode in snap.iterStormPodes(text, opts=opts, user=user): await chan.put(('node', pode)) count += 1 else: info = opts.get('_loginfo', {}) info.update({'mode': opts.get('mode', 'storm'), 'view': self.iden}) self.core._logStormQuery(text, user, info=info) async for item in snap.storm(text, opts=opts, user=user): count += 1 except s_stormctrl.StormExit: pass except asyncio.CancelledError: logger.warning('Storm runtime cancelled.', extra={'synapse': {'text': text, 'username': user.name, 'user': user.iden}}) cancelled = True raise except Exception as e: mesg = '' if isinstance(e, s_stormctrl.StormLoopCtrl): mesg = f'Loop control statement "{e.statement}" used outside of a loop.' e = s_exc.StormRuntimeError(mesg=mesg, statement=e.statement, highlight=e.get('highlight')) elif isinstance(e, s_stormctrl.StormGenrCtrl): mesg = f'Generator control statement "{e.statement}" used outside of a generator function.' e = s_exc.StormRuntimeError(mesg=mesg, statement=e.statement, highlight=e.get('highlight')) logmesg = f'Error during storm execution for {{ {text} }}' if mesg: logmesg = f'{logmesg} - {mesg}' logger.exception(logmesg, extra={'synapse': {'text': text, 'username': user.name, 'user': user.iden}}) enfo = s_common.err(e) enfo[1].pop('esrc', None) enfo[1].pop('ename', None) await chan.put(('err', enfo)) finally: if not cancelled: abstock = s_common.mononow() abstook = abstock - abstick tock = tick + abstook await chan.put(('fini', {'tock': tock, 'abstock': abstock, 'took': abstook, 'count': count, })) await synt.worker(runStorm(), name='runstorm') editformat = opts.get('editformat', 'nodeedits') while True: mesg = await chan.get() kind = mesg[0] if kind == 'node': yield mesg continue if kind == 'node:edits': if editformat == 'nodeedits': nodeedits = s_common.jsonsafe_nodeedits(mesg[1]['edits']) mesg[1]['edits'] = nodeedits yield mesg continue if editformat == 'none': continue assert editformat == 'count' count = sum(len(edit[2]) for edit in mesg[1].get('edits', ())) mesg = ('node:edits:count', {'count': count}) yield mesg continue if kind == 'fini': yield mesg break yield mesg
[docs] async def iterStormPodes(self, text, opts=None): opts = self.core._initStormOpts(opts) user = self.core._userFromOpts(opts) taskinfo = {'query': text, 'view': self.iden} taskiden = opts.get('task') await self.core.boss.promote('storm', user=user, info=taskinfo, taskiden=taskiden) with s_scope.enter({'user': user}): async with await self.snap(user=user) as snap: async for pode in snap.iterStormPodes(text, opts=opts, user=user): yield pode
[docs] async def snap(self, user): if self.invalid is not None: raise s_exc.NoSuchLayer(mesg=f'No such layer {self.invalid}', iden=self.invalid) return await self.snapctor(self, user)
[docs] @s_nexus.Pusher.onPushAuto('trig:q:add', passitem=True) async def addTrigQueue(self, triginfo, nexsitem): nexsoff, nexsmesg = nexsitem self.trigqueue.add(triginfo, indx=nexsoff)
[docs] @s_nexus.Pusher.onPushAuto('trig:q:del') async def delTrigQueue(self, offs): self.trigqueue.pop(offs)
[docs] @s_nexus.Pusher.onPushAuto('view:set') async def setViewInfo(self, name, valu): ''' Set a mutable view property. ''' if name not in ('name', 'desc', 'parent', 'nomerge', 'protected', 'quorum'): # TODO: Remove nomerge after Synapse 3.x.x mesg = f'{name} is not a valid view info key' raise s_exc.BadOptValu(mesg=mesg) if name == 'parent': parent = self.core.reqView(valu, mesg='The parent view must already exist.') if parent.iden == self.iden: mesg = 'A view may not have parent set to itself.' raise s_exc.BadArg(mesg=mesg) if parent.isForkOf(self.iden): mesg = 'Circular dependency of view parents is not supported.' raise s_exc.BadArg(mesg=mesg) if self.parent is not None: if self.parent.iden == parent.iden: return valu mesg = 'You may not set parent on a view which already has one.' raise s_exc.BadArg(mesg=mesg) if len(self.layers) != 1: mesg = 'You may not set parent on a view which has more than one layer.' raise s_exc.BadArg(mesg=mesg) self.parent = parent self.info[name] = valu self.core.viewdefs.set(self.iden, self.info) await self._calcForkLayers() for view in self.core.views.values(): if view.isForkOf(self.iden): await view._calcForkLayers() self.core._calcViewsByLayer() else: if name == 'quorum': # TODO hack a schema test until the setViewInfo API is updated to # enforce ( which will need to be done very carefully to prevent # existing non-compliant values from causing issues with existing views ) if valu is not None: vdef = s_msgpack.deepcopy(self.info) vdef['quorum'] = s_msgpack.deepcopy(valu) s_schemas.reqValidView(vdef) else: for view in self.core.views.values(): if view.parent != self: continue if view.getMergeRequest() is not None: await view._delMergeRequest() if valu is None: self.info.pop(name, None) else: self.info[name] = valu self.core.viewdefs.set(self.iden, self.info) await self.core.feedBeholder('view:set', {'iden': self.iden, 'name': name, 'valu': valu}, gates=[self.iden, self.layers[0].iden]) return valu
async def _setLayerIdens(self, idens): # this may only be called from within a nexus handler... self.info['layers'] = idens self.core.viewdefs.set(self.iden, self.info) await self._initViewLayers() await self.core.feedBeholder('view:set', {'iden': self.iden, 'name': 'layers', 'valu': idens}, gates=[self.iden, self.layers[0].iden])
[docs] async def addLayer(self, layriden, indx=None): if any(layriden == layr.iden for layr in self.layers): raise s_exc.DupIden(mesg='May not have the same layer in a view twice') return await self._push('view:addlayer', layriden, indx)
@s_nexus.Pusher.onPush('view:addlayer') async def _addLayer(self, layriden, indx=None): for view in self.core.views.values(): if view.parent is self: raise s_exc.ReadOnlyLayer(mesg='May not change layers that have been forked from') if self.parent is not None: raise s_exc.ReadOnlyLayer(mesg='May not change layers of forked view') layr = self.core.layers.get(layriden) if layr is None: raise s_exc.NoSuchLayer(mesg=f'No such layer {layriden}', iden=layriden) if layr in self.layers: return if indx is None: self.layers.append(layr) else: self.layers.insert(indx, layr) self.info['layers'] = [lyr.iden for lyr in self.layers] self.core.viewdefs.set(self.iden, self.info) await self.core.feedBeholder('view:addlayer', {'iden': self.iden, 'layer': layriden, 'indx': indx}, gates=[self.iden, layriden]) self.core._calcViewsByLayer()
[docs] @s_nexus.Pusher.onPushAuto('view:setlayers') async def setLayers(self, layers): ''' Set the view layers from a list of idens. NOTE: view layers are stored "top down" (the write layer is self.layers[0]) ''' layrs = [] if self.parent is not None: mesg = 'You cannot set the layers of a forked view.' raise s_exc.BadArg(mesg=mesg) for iden in layers: layr = self.core.layers.get(iden) if layr is None: raise s_exc.NoSuchLayer(mesg=f'No such layer {iden}', iden=iden) if not layrs and layr.readonly: raise s_exc.ReadOnlyLayer(mesg=f'First layer {layr.iden} must not be read-only') layrs.append(layr) self.invalid = None self.layers = layrs self.info['layers'] = layers self.core.viewdefs.set(self.iden, self.info) await self.core.feedBeholder('view:setlayers', {'iden': self.iden, 'layers': layers}, gates=[self.iden, layers[0]]) await self._calcChildViews() self.core._calcViewsByLayer()
async def _calcChildViews(self): todo = collections.deque([self]) byparent = collections.defaultdict(list) for view in self.core.views.values(): if view.parent is None: continue byparent[view.parent].append(view) while todo: view = todo.pop() for child in byparent.get(view, ()): layers = [child.layers[0]] layers.extend(view.layers) child.layers = layers # convert layers to a list of idens... lids = [layr.iden for layr in layers] child.info['layers'] = lids self.core.viewdefs.set(child.iden, child.info) await self.core.feedBeholder('view:setlayers', {'iden': child.iden, 'layers': lids}, gates=[child.iden, lids[0]]) todo.append(child)
[docs] async def children(self): for view in list(self.core.views.values()): if view.parent != self: await asyncio.sleep(0) continue yield view
[docs] async def insertParentFork(self, useriden, name=None): ''' Insert a new View between a forked View and its parent. Returns: New view definition with the same perms as the current fork. ''' if not self.isafork(): mesg = f'View ({self.iden}) is not a fork, cannot insert a new fork between it and parent.' raise s_exc.BadState(mesg=mesg) ctime = s_common.now() layriden = s_common.guid() ldef = { 'iden': layriden, 'created': ctime, 'creator': useriden, 'lockmemory': self.core.conf.get('layers:lockmemory'), 'logedits': self.core.conf.get('layers:logedits'), 'readonly': False } vdef = { 'iden': s_common.guid(), 'created': ctime, 'creator': useriden, 'parent': self.parent.iden, 'layers': [layriden] + [lyr.iden for lyr in self.parent.layers] } if name is not None: vdef['name'] = name s_layer.reqValidLdef(ldef) s_schemas.reqValidView(vdef) return await self._push('view:forkparent', ldef, vdef)
@s_nexus.Pusher.onPush('view:forkparent', passitem=True) async def _insertParentFork(self, ldef, vdef, nexsitem): s_layer.reqValidLdef(ldef) s_schemas.reqValidView(vdef) if self.getMergeRequest() is not None: await self._delMergeRequest() await self.core._addLayer(ldef, nexsitem) await self.core._addView(vdef) forkiden = vdef.get('iden') self.info['parent'] = forkiden self.parent = self.core.reqView(forkiden) self.core.viewdefs.set(self.iden, self.info) mesg = {'iden': self.iden, 'name': 'parent', 'valu': forkiden} await self.core.feedBeholder('view:set', mesg, gates=[self.iden, self.layers[0].iden]) await self._calcForkLayers() for view in self.core.views.values(): if view.isForkOf(self.iden): await view._calcForkLayers() self.core._calcViewsByLayer() authgate = await self.core.getAuthGate(self.iden) if authgate is None: # pragma: no cover return await self.parent.pack() for userinfo in authgate.get('users'): useriden = userinfo.get('iden') if (user := self.core.auth.user(useriden)) is None: # pragma: no cover logger.warning(f'View {self.iden} AuthGate refers to unknown user {useriden}') continue await user.setRules(userinfo.get('rules'), gateiden=forkiden, nexs=False) await user.setAdmin(userinfo.get('admin'), gateiden=forkiden, logged=False) for roleinfo in authgate.get('roles'): roleiden = roleinfo.get('iden') if (role := self.core.auth.role(roleiden)) is None: # pragma: no cover logger.warning(f'View {self.iden} AuthGate refers to unknown role {roleiden}') continue await role.setRules(roleinfo.get('rules'), gateiden=forkiden, nexs=False) return await self.parent.pack()
[docs] async def fork(self, ldef=None, vdef=None): ''' Make a new view inheriting from this view with the same layers and a new write layer on top Args: ldef: layer parameter dict vdef: view parameter dict Passed through to cortex.addLayer Returns: new view object, with an iden the same as the new write layer iden ''' if ldef is None: ldef = {} if vdef is None: vdef = {} ldef = await self.core.addLayer(ldef) layriden = ldef.get('iden') vdef['parent'] = self.iden vdef['layers'] = [layriden] + [lyr.iden for lyr in self.layers] return await self.core.addView(vdef)
[docs] async def merge(self, useriden=None, force=False): ''' Merge this view into its parent. All changes made to this view will be applied to the parent. Parent's triggers will be run. ''' fromlayr = self.layers[0] if useriden is None: user = await self.core.auth.getUserByName('root') else: user = await self.core.auth.reqUser(useriden) await self.mergeAllowed(user, force=force) taskinfo = {'merging': self.iden, 'view': self.iden} await self.core.boss.promote('storm', user=user, info=taskinfo) async with await self.parent.snap(user=user) as snap: meta = await snap.getSnapMeta() async for nodeedits in fromlayr.iterLayerNodeEdits(): await self.parent.storNodeEdits([nodeedits], meta)
[docs] async def swapLayer(self): oldlayr = self.layers[0] newlayr = await self.core._twinLayer(oldlayr) await self.core.swapLayer(oldlayr.iden, newlayr.iden) await self.core.delLayer(oldlayr.iden)
[docs] async def wipeLayer(self, useriden=None): ''' Delete the data in the write layer by generating del nodeedits. Triggers will be run. ''' if useriden is None: user = await self.core.auth.getUserByName('root') else: user = await self.core.auth.reqUser(useriden) await self.wipeAllowed(user) async with await self.snap(user=user) as snap: meta = await snap.getSnapMeta() async for nodeedit in self.layers[0].iterWipeNodeEdits(): await snap.getNodeByBuid(nodeedit[0]) # to load into livenodes for callbacks await snap.saveNodeEdits([nodeedit], meta)
def _confirm(self, user, perms): layriden = self.layers[0].iden if user.allowed(perms, gateiden=layriden): return perm = '.'.join(perms) mesg = f'User ({user.name}) must have permission {perm} on write layer {layriden} of view {self.iden}' raise s_exc.AuthDeny(mesg=mesg, perm=perm, user=user.iden, username=user.name)
[docs] async def mergeAllowed(self, user=None, force=False): ''' Check whether a user can merge a view into its parent. NOTE: This API may not be used to check for merges based on quorum votes. ''' fromlayr = self.layers[0] if self.parent is None: raise s_exc.CantMergeView(mesg=f'Cannot merge view ({self.iden}) that has not been forked.') if self.info.get('protected'): raise s_exc.CantMergeView(mesg=f'Cannot merge view ({self.iden}) that has protected set.') if self.parent.info.get('quorum') is not None: raise s_exc.CantMergeView(mesg=f'Cannot merge view({self.iden}). Parent view requires quorum voting.') if self.trigqueue.size and not force: raise s_exc.CantMergeView(mesg=f'There are still {self.trigqueue.size} triggers waiting to complete.', canforce=True) parentlayr = self.parent.layers[0] if parentlayr.readonly: raise s_exc.ReadOnlyLayer(mesg="May not merge if the parent's write layer is read-only") for view in self.core.views.values(): if view.parent == self: raise s_exc.CantMergeView(mesg='Cannot merge a view that has children itself') if user is None or user.isAdmin() or user.isAdmin(gateiden=parentlayr.iden): return await fromlayr.confirmLayerEditPerms(user, parentlayr.iden)
[docs] async def wipeAllowed(self, user=None): ''' Check whether a user can wipe the write layer in the current view. ''' if user is None or user.isAdmin(): return layer = self.layers[0] await layer.confirmLayerEditPerms(user, layer.iden, delete=True)
[docs] async def runTagAdd(self, node, tag, valu): if self.core.migration: return # Run any trigger handlers await self.triggers.runTagAdd(node, tag)
[docs] async def runTagDel(self, node, tag, valu): if self.core.migration: return await self.triggers.runTagDel(node, tag)
[docs] async def runNodeAdd(self, node): if self.core.migration: return await self.triggers.runNodeAdd(node)
[docs] async def runNodeDel(self, node): if self.core.migration: return await self.triggers.runNodeDel(node)
[docs] async def runPropSet(self, node, prop, oldv): ''' Handle when a prop set trigger event fired ''' if self.core.migration: return await self.triggers.runPropSet(node, prop, oldv)
[docs] async def runEdgeAdd(self, n1, edge, n2): if self.core.migration: return await self.triggers.runEdgeAdd(n1, edge, n2)
[docs] async def runEdgeDel(self, n1, edge, n2): if self.core.migration: return await self.triggers.runEdgeDel(n1, edge, n2)
[docs] async def addTrigger(self, tdef): ''' Adds a trigger to the view. ''' iden = tdef.get('iden') if iden is None: tdef['iden'] = s_common.guid() elif self.triggers.get(iden) is not None: raise s_exc.DupIden(mesg='A trigger with this iden already exists') tdef['view'] = self.iden root = await self.core.auth.getUserByName('root') tdef.setdefault('created', s_common.now()) tdef.setdefault('user', root.iden) tdef.setdefault('async', False) tdef.setdefault('enabled', True) s_trigger.reqValidTdef(tdef) return await self._push('trigger:add', tdef)
@s_nexus.Pusher.onPush('trigger:add') async def _onPushAddTrigger(self, tdef): s_trigger.reqValidTdef(tdef) trig = self.trigdict.get(tdef['iden']) if trig is not None: return self.triggers.get(tdef['iden']).pack() self.core.auth.reqNoAuthGate(tdef['iden']) user = self.core.auth.user(tdef['user']) await self.core.getStormQuery(tdef['storm']) trig = await self.triggers.load(tdef) self.trigdict.set(trig.iden, tdef) await self.core.auth.addAuthGate(trig.iden, 'trigger') await user.setAdmin(True, gateiden=tdef.get('iden'), logged=False) await self.core.feedBeholder('trigger:add', trig.pack(), gates=[trig.iden]) return trig.pack()
[docs] async def getTrigger(self, iden): trig = self.triggers.get(iden) if trig is None: raise s_exc.NoSuchIden(mesg=f"Trigger not found {iden=}", iden=iden) return trig
[docs] async def delTrigger(self, iden): trig = self.triggers.get(iden) if trig is None: raise s_exc.NoSuchIden(mesg=f"Trigger not found {iden=}", iden=iden) return await self._push('trigger:del', iden)
@s_nexus.Pusher.onPush('trigger:del') async def _delTrigger(self, iden): ''' Delete a trigger from the view. ''' trig = self.triggers.pop(iden) if trig is None: return await self.core.feedBeholder('trigger:del', {'iden': trig.iden, 'view': trig.view.iden}, gates=[trig.iden]) self.trigdict.pop(trig.iden) await self.core.auth.delAuthGate(trig.iden)
[docs] @s_nexus.Pusher.onPushAuto('trigger:set') async def setTriggerInfo(self, iden, name, valu): trig = self.triggers.get(iden) if trig is None: raise s_exc.NoSuchIden(mesg=f"Trigger not found {iden=}", iden=iden) await trig.set(name, valu) await self.core.feedBeholder('trigger:set', {'iden': trig.iden, 'view': trig.view.iden, 'name': name, 'valu': valu}, gates=[trig.iden])
[docs] async def listTriggers(self): ''' List all the triggers in the view. ''' trigs = self.triggers.list() return trigs
[docs] async def delete(self): ''' Delete the metadata for this view. Note: this does not delete any layer storage. ''' await self.fini() await self.trigdict.truncate() await self._wipeViewMeta() shutil.rmtree(self.dirn, ignore_errors=True)
[docs] async def addNode(self, form, valu, props=None, user=None): async with await self.snap(user=user) as snap: return await snap.addNode(form, valu, props=props)
[docs] async def addNodeEdits(self, edits, meta): ''' A telepath compatible way to apply node edits to a view. NOTE: This does cause trigger execution. ''' user = await self.core.auth.reqUser(meta.get('user')) async with await self.snap(user=user) as snap: # go with the anti-pattern for now... await snap.saveNodeEdits(edits, None)
[docs] async def storNodeEdits(self, edits, meta): return await self.addNodeEdits(edits, meta)
# TODO remove addNodeEdits?
[docs] async def scrapeIface(self, text, unique=False, refang=True): async with await s_spooled.Set.anit(dirn=self.core.dirn, cell=self.core) as matches: # type: s_spooled.Set # The synapse.lib.scrape APIs handle form arguments for us. async for item in s_scrape.contextScrapeAsync(text, refang=refang, first=False): form = item.pop('form') valu = item.pop('valu') if unique: key = (form, valu) if key in matches: await asyncio.sleep(0) continue await matches.add(key) try: tobj = self.core.model.type(form) valu, _ = tobj.norm(valu) except s_exc.BadTypeValu: await asyncio.sleep(0) continue # Yield a tuple of <form, normed valu, info> yield form, valu, item # Return early if the scrape interface is disabled if not self.core.stormiface_scrape: return # Scrape interface: # # The expected scrape interface takes a text and optional form # argument. # # The expected interface implementation returns a list/tuple of # (form, valu, info) results. Info is expected to contain the # match offset and raw valu. # # Scrape implementers are responsible for ensuring that their # resulting match and offsets are found in the text we sent # to them. todo = s_common.todo('scrape', text) async for results in self.callStormIface('scrape', todo): for (form, valu, info) in results: if unique: key = (form, valu) if key in matches: await asyncio.sleep(0) continue await matches.add(key) try: tobj = self.core.model.type(form) valu, _ = tobj.norm(valu) except AttributeError: # pragma: no cover logger.exception(f'Scrape interface yielded unknown form {form}') await asyncio.sleep(0) continue except (s_exc.BadTypeValu): # pragma: no cover await asyncio.sleep(0) continue # Yield a tuple of <form, normed valu, info> yield form, valu, info await asyncio.sleep(0)