2017-10-26 08:17:17 +00:00
|
|
|
import os, re, base64, zlib, hashlib, binascii
|
2015-08-14 10:12:10 +00:00
|
|
|
import xml.etree.ElementTree as ET
|
|
|
|
|
|
|
|
PY_SEP = os.path.sep
|
|
|
|
|
|
|
|
def wrapScope( src ):
|
|
|
|
return "(function(){" + src + "})();"
|
|
|
|
|
|
|
|
class Resolver:
|
|
|
|
resolved = None
|
|
|
|
|
|
|
|
bMap = None
|
|
|
|
parentMap = None
|
|
|
|
|
|
|
|
EX_PROP = "prop"
|
|
|
|
EX_FUNC = "method"
|
|
|
|
EX_CLASS = "class"
|
|
|
|
|
2022-08-02 14:46:44 +00:00
|
|
|
_rLookup = None
|
|
|
|
|
2015-08-14 10:12:10 +00:00
|
|
|
def __init__( self, classMap ):
|
|
|
|
self.classMap = classMap
|
2022-08-02 14:46:44 +00:00
|
|
|
self._rLookup = {}
|
2015-08-14 10:12:10 +00:00
|
|
|
self._reload()
|
|
|
|
|
|
|
|
def _reload( self ):
|
|
|
|
self.bMap = ET.parse( self.classMap )
|
|
|
|
# ElementTree does not support Element.find( ".." )
|
|
|
|
# Make a parent map to work around
|
|
|
|
self.parentMap = { c:p for p in self.bMap.iter() for c in p }
|
|
|
|
|
|
|
|
self.resolved = []
|
|
|
|
|
|
|
|
def resource( self, elem ):
|
|
|
|
if "src" in elem.attrib:
|
2022-08-02 14:46:44 +00:00
|
|
|
key = elem.attrib[ "src" ]
|
|
|
|
if not key in self._rLookup:
|
|
|
|
self._rLookup[ key ] = {
|
|
|
|
"src": elem.attrib[ "src" ]
|
|
|
|
, "js": elem.attrib[ "js" ]
|
|
|
|
, "css": elem.attrib[ "css" ]
|
|
|
|
}
|
|
|
|
return self._rLookup[ key ]
|
2015-08-14 10:12:10 +00:00
|
|
|
|
|
|
|
parent = self.parentMap[ elem ]
|
|
|
|
|
2022-08-02 16:20:17 +00:00
|
|
|
if parent is not None:
|
2015-08-14 10:12:10 +00:00
|
|
|
return self.resource( parent )
|
|
|
|
|
2022-08-02 14:46:44 +00:00
|
|
|
def locate( self, key ):
|
|
|
|
return self._rLookup.get( key )
|
|
|
|
|
2015-08-14 10:12:10 +00:00
|
|
|
def resolve( self, c, classList ):
|
|
|
|
self.resolved = []
|
|
|
|
self.__resolve( c, classList )
|
|
|
|
|
|
|
|
def __resolve( self, c, classList ):
|
|
|
|
|
|
|
|
lista = list( "[@name=\"" + x + "\"]" for x in c.split(".") )
|
|
|
|
|
|
|
|
fx = "/".join( self.EX_CLASS + x for x in lista[:-1] )
|
|
|
|
|
2016-03-14 19:01:46 +00:00
|
|
|
# resolve wildcard A.B.*
|
|
|
|
if c[-2:] == ".*":
|
|
|
|
elem = self.bMap.findall( ".//" + fx + self.EX_CLASS )
|
|
|
|
|
|
|
|
if elem == None:
|
|
|
|
raise LookupError( "Namespace does not exists or contains no classes: " + c )
|
|
|
|
|
|
|
|
c = c[0:-1]
|
|
|
|
for cl in elem:
|
|
|
|
cl = c + cl.attrib[ "name" ]
|
|
|
|
|
|
|
|
if cl not in self.resolved:
|
|
|
|
self.__resolve( cl, classList )
|
|
|
|
|
|
|
|
return
|
|
|
|
|
2015-08-14 10:12:10 +00:00
|
|
|
it = lista[-1]
|
|
|
|
# Test if class
|
|
|
|
elem = self.bMap.find( ".//" + fx + self.EX_CLASS + it )
|
|
|
|
|
|
|
|
if elem == None:
|
|
|
|
if fx != '': fx += "/"
|
|
|
|
|
|
|
|
# Test if prop
|
|
|
|
elem = self.bMap.find( ".//" + fx + self.EX_PROP + it )
|
|
|
|
|
|
|
|
# test if func
|
|
|
|
if elem == None:
|
|
|
|
elem = self.bMap.find( ".//" + fx + self.EX_FUNC + it )
|
|
|
|
|
|
|
|
if elem == None:
|
|
|
|
raise LookupError( "No such class: " + c )
|
|
|
|
|
|
|
|
imports = self.parentMap[ elem ].findall( "import" )
|
|
|
|
|
|
|
|
else:
|
|
|
|
imports = elem.findall( "import" )
|
|
|
|
|
|
|
|
self.resolved.append( c )
|
|
|
|
|
|
|
|
for imp in imports:
|
|
|
|
if imp.text not in self.resolved:
|
|
|
|
self.__resolve( imp.text, classList )
|
|
|
|
|
|
|
|
classList.append([ c, elem ])
|
|
|
|
|
|
|
|
class BotanClassResolver:
|
|
|
|
R = ""
|
|
|
|
CR = None
|
|
|
|
|
|
|
|
classMap = ""
|
|
|
|
flagCompress = True
|
2021-09-05 15:16:37 +00:00
|
|
|
oModeIfSizelt = 50 * 1024
|
2015-08-14 10:12:10 +00:00
|
|
|
returnHash = False
|
2021-09-05 15:16:37 +00:00
|
|
|
returnDynamic = False
|
2015-08-14 10:12:10 +00:00
|
|
|
resv = None
|
|
|
|
|
|
|
|
def __init__( self, jwork, BotanRoot, classMap, cacheRoot ):
|
|
|
|
|
|
|
|
self.JWork = jwork;
|
|
|
|
self.R = os.path.abspath( BotanRoot )
|
|
|
|
self.CR = os.path.abspath( cacheRoot )
|
|
|
|
|
|
|
|
os.makedirs( self.CR, 0o755, exist_ok = True )
|
|
|
|
|
|
|
|
if not os.path.exists( classMap ):
|
|
|
|
self.JWork.buildClassMap( self.R, classMap )
|
|
|
|
|
|
|
|
self.resv = Resolver( classMap )
|
|
|
|
|
|
|
|
def BotanFile( self, t ):
|
|
|
|
content = ""
|
|
|
|
with open( os.path.join( self.R, t ), "r" ) as f:
|
|
|
|
content = f.read()
|
|
|
|
|
|
|
|
return content
|
|
|
|
|
2021-09-05 15:16:37 +00:00
|
|
|
def BotanCache( self, srcFile, fileHash, hashContentDynamic ):
|
2015-08-14 10:12:10 +00:00
|
|
|
|
2021-09-05 15:16:37 +00:00
|
|
|
for _ in [0]:
|
|
|
|
if hashContentDynamic and os.path.getsize( srcFile ) < self.oModeIfSizelt:
|
|
|
|
break
|
|
|
|
|
|
|
|
if self.returnHash:
|
|
|
|
return fileHash
|
|
|
|
|
|
|
|
with open( srcFile, "r" ) as f:
|
|
|
|
return f.read()
|
2015-08-14 10:12:10 +00:00
|
|
|
|
|
|
|
def cleanList( self, lista ):
|
|
|
|
olist = []
|
|
|
|
for i in lista:
|
|
|
|
if i not in olist:
|
|
|
|
olist.append( i )
|
|
|
|
return olist
|
|
|
|
|
|
|
|
def jsLookup( self, classList, classFiles ):
|
|
|
|
for c in classList:
|
|
|
|
src = self.resv.resource( c[1] )
|
|
|
|
|
|
|
|
if src == None:
|
|
|
|
raise LookupError( "Cannot find src file for: " + c[0] )
|
|
|
|
|
|
|
|
if src not in classFiles:
|
|
|
|
classFiles.append( src )
|
|
|
|
|
2022-08-02 13:35:37 +00:00
|
|
|
def cssLookup( self, classList, cssList ):
|
2015-08-14 10:12:10 +00:00
|
|
|
|
2022-08-02 13:35:37 +00:00
|
|
|
for cdef in classList:
|
|
|
|
f = cdef[ "src" ]
|
2015-08-14 10:12:10 +00:00
|
|
|
|
|
|
|
cssFile = os.path.splitext( f )[0] + ".css"
|
|
|
|
|
2022-08-02 15:17:51 +00:00
|
|
|
cssGroup = []
|
2022-08-02 14:46:44 +00:00
|
|
|
if not cdef[ "css" ] == "1":
|
2022-08-02 15:17:51 +00:00
|
|
|
cssGroup.append( cdef )
|
2015-08-14 10:12:10 +00:00
|
|
|
|
|
|
|
f = f.split( PY_SEP )
|
|
|
|
l = len( f )
|
|
|
|
|
|
|
|
for i in range( 1, l ):
|
2022-08-02 14:46:44 +00:00
|
|
|
key = PY_SEP.join( x for x in f[:-i] ) + PY_SEP + "_this.js"
|
|
|
|
_def = self.resv.locate( key )
|
2015-08-14 10:12:10 +00:00
|
|
|
|
2022-08-02 14:46:44 +00:00
|
|
|
if _def and not _def[ "css" ] == "1":
|
2022-08-02 15:17:51 +00:00
|
|
|
cssGroup.append( _def )
|
2015-08-14 10:12:10 +00:00
|
|
|
|
2022-08-02 15:17:51 +00:00
|
|
|
cssGroup.sort( key = lambda x : x[ "src" ] )
|
|
|
|
cssList.extend( cssGroup )
|
|
|
|
|
2015-08-14 10:12:10 +00:00
|
|
|
def getCache( self, fileList, cName, mode ):
|
|
|
|
if self.CR == None:
|
|
|
|
return None
|
|
|
|
|
2022-08-02 13:35:37 +00:00
|
|
|
md5 = hashlib.md5( bytearray( "|".join( x[mode] for x in fileList ), "utf-8" ) ).hexdigest()
|
2015-08-14 10:12:10 +00:00
|
|
|
|
|
|
|
cName[0] = oFHash = md5 + "." + mode
|
|
|
|
cFHash = md5 + ".c." + mode
|
|
|
|
|
|
|
|
# Raw file
|
|
|
|
oFile = os.path.join( self.CR, oFHash )
|
|
|
|
# Compressed file
|
|
|
|
cFile = os.path.join( self.CR, cFHash )
|
|
|
|
|
2022-08-02 16:20:17 +00:00
|
|
|
cCached = self.useCache( cFile )
|
|
|
|
|
|
|
|
if self.flagCompress and cCached:
|
2021-09-05 15:16:37 +00:00
|
|
|
return self.BotanCache( cFile, cFHash, self.returnDynamic )
|
2015-08-14 10:12:10 +00:00
|
|
|
|
2022-08-02 16:20:17 +00:00
|
|
|
if self.useCache( oFile ):
|
|
|
|
|
|
|
|
if not cCached:
|
|
|
|
self.JWork.saveCache(
|
|
|
|
oFile
|
|
|
|
# Content is None to initiate a compression
|
|
|
|
, None
|
|
|
|
, mode
|
|
|
|
, os.path.join( self.R, "externs" )
|
|
|
|
)
|
2015-08-14 10:12:10 +00:00
|
|
|
|
2021-09-05 15:16:37 +00:00
|
|
|
return self.BotanCache( oFile, oFHash, False )
|
2015-08-14 10:12:10 +00:00
|
|
|
|
2022-08-02 16:20:17 +00:00
|
|
|
return None
|
|
|
|
|
2022-08-02 13:35:37 +00:00
|
|
|
def useCache( self, f ):
|
|
|
|
return os.path.exists( f )
|
2015-08-14 10:12:10 +00:00
|
|
|
|
|
|
|
def compileJs( self, cList, xList ):
|
|
|
|
md5 = [ None ]
|
|
|
|
|
|
|
|
for x in xList:
|
|
|
|
cList.remove( x ) if x in cList else None
|
|
|
|
|
|
|
|
cacheFile = self.getCache( cList, md5, "js" )
|
|
|
|
|
2022-08-02 16:20:17 +00:00
|
|
|
if cacheFile is not None:
|
2015-08-14 10:12:10 +00:00
|
|
|
return cacheFile;
|
|
|
|
|
|
|
|
# The root file
|
|
|
|
outputJs = self.BotanFile( "_this.js" )
|
|
|
|
|
|
|
|
for f in cList:
|
2022-08-02 13:35:37 +00:00
|
|
|
f = f[ "src" ]
|
2015-08-14 10:12:10 +00:00
|
|
|
path = (
|
|
|
|
os.path.splitext( f )[0]
|
|
|
|
.replace( PY_SEP, "." )
|
|
|
|
.replace( "._this", "" )
|
|
|
|
)
|
|
|
|
struct = ";BotanJS.define( \"" + path + "\" );"
|
|
|
|
|
|
|
|
outputJs += struct + self.BotanFile( f )
|
|
|
|
|
|
|
|
outputJs = wrapScope( outputJs )
|
|
|
|
|
2021-09-05 15:16:37 +00:00
|
|
|
self.JWork.saveCache(
|
2015-08-14 10:12:10 +00:00
|
|
|
os.path.join( self.CR, md5[0] )
|
|
|
|
, outputJs
|
|
|
|
, "js"
|
|
|
|
, os.path.join( self.R, "externs" )
|
|
|
|
)
|
|
|
|
|
|
|
|
if self.returnHash:
|
|
|
|
return md5[0]
|
|
|
|
|
|
|
|
return outputJs
|
|
|
|
|
|
|
|
def compileCss( self, cList, xList ):
|
|
|
|
cssIList = []
|
|
|
|
cssXList = []
|
|
|
|
self.cssLookup( cList, cssIList )
|
|
|
|
self.cssLookup( xList, cssXList )
|
|
|
|
|
|
|
|
cList = []
|
|
|
|
xList = cssXList
|
|
|
|
|
|
|
|
for x in cssIList:
|
|
|
|
cList.append( x ) if x not in xList else None
|
|
|
|
|
|
|
|
md5 = [ None ]
|
|
|
|
cacheFile = self.getCache( cList, md5, "css" )
|
|
|
|
|
2022-08-02 16:20:17 +00:00
|
|
|
if cacheFile is not None:
|
2015-08-14 10:12:10 +00:00
|
|
|
return cacheFile;
|
|
|
|
|
2022-08-02 13:35:37 +00:00
|
|
|
struct = "/* @ */"
|
|
|
|
outputCss = struct + self.BotanFile( "_this.css" )
|
2015-08-14 10:12:10 +00:00
|
|
|
|
|
|
|
for f in self.cleanList( cList ):
|
2022-08-02 14:46:44 +00:00
|
|
|
outputCss += self.BotanFile( f["src"][:-2] + "css" )
|
2015-08-14 10:12:10 +00:00
|
|
|
|
2021-09-05 15:16:37 +00:00
|
|
|
self.JWork.saveCache( os.path.join( self.CR, md5[0] ), outputCss, "css" )
|
2015-08-14 10:12:10 +00:00
|
|
|
|
|
|
|
if self.returnHash:
|
|
|
|
return md5[0]
|
|
|
|
|
|
|
|
return outputCss
|
|
|
|
|
|
|
|
def getAPI( self, code, mode ):
|
|
|
|
self.flagCompress = True
|
|
|
|
# Should reload on debug mode only
|
|
|
|
self.resv._reload()
|
|
|
|
flag = mode[0]
|
|
|
|
requestAPIs = code
|
|
|
|
|
2021-09-05 15:16:37 +00:00
|
|
|
# Return compressed contents if possible
|
|
|
|
# otherwise return raw contents
|
2015-08-14 10:12:10 +00:00
|
|
|
if flag == "o":
|
|
|
|
mode = mode[1:]
|
2021-09-05 15:16:37 +00:00
|
|
|
|
|
|
|
# Return raw contents only
|
2015-08-14 10:12:10 +00:00
|
|
|
elif flag == "r":
|
|
|
|
mode = mode[1:]
|
|
|
|
self.flagCompress = False
|
2021-09-05 15:16:37 +00:00
|
|
|
|
|
|
|
# Return hashed filenames only
|
|
|
|
elif flag == "h":
|
|
|
|
mode = mode[1:]
|
|
|
|
self.returnHash = True
|
|
|
|
|
|
|
|
# Return hashed filenames if content is larger than self.oModeIfSizelt bytes
|
|
|
|
# otherwise act as "o" mode
|
2015-08-14 10:12:10 +00:00
|
|
|
else:
|
|
|
|
self.returnHash = True
|
2021-09-05 15:16:37 +00:00
|
|
|
self.returnDynamic = True
|
2017-10-26 08:17:17 +00:00
|
|
|
|
|
|
|
try:
|
2015-08-14 10:12:10 +00:00
|
|
|
requestAPIs = (
|
|
|
|
# decode -> decompress -> split
|
|
|
|
zlib.decompress( base64.b64decode( code, None, True ) )
|
|
|
|
.decode( "utf-8" )
|
|
|
|
)
|
|
|
|
sp = ","
|
2017-10-26 08:17:17 +00:00
|
|
|
except binascii.Error:
|
|
|
|
sp = "/"
|
2015-08-14 10:12:10 +00:00
|
|
|
|
|
|
|
# strip malicious
|
|
|
|
requestAPIs = (
|
|
|
|
requestAPIs
|
|
|
|
.replace( "[^A-Za-z\.\*" + re.escape( sp ) + " ]", "" )
|
|
|
|
.split( sp )
|
|
|
|
)
|
|
|
|
|
|
|
|
imports = []
|
|
|
|
excludes = []
|
|
|
|
|
|
|
|
for apis in requestAPIs:
|
|
|
|
|
2021-09-05 15:16:37 +00:00
|
|
|
if not apis:
|
|
|
|
continue
|
2015-08-14 10:12:10 +00:00
|
|
|
|
|
|
|
classList = []
|
|
|
|
lookupList = imports
|
|
|
|
|
|
|
|
if apis[0] == "-":
|
|
|
|
apis = apis[1:]
|
|
|
|
lookupList = excludes
|
|
|
|
|
|
|
|
self.resv.resolve( apis, classList )
|
|
|
|
self.jsLookup( classList, lookupList )
|
|
|
|
|
|
|
|
if mode == "js":
|
|
|
|
return self.compileJs( imports, excludes )
|
|
|
|
elif mode == "css":
|
|
|
|
return self.compileCss( imports, excludes )
|
|
|
|
|
|
|
|
raise TypeError( "Invalid mode: " + js )
|