Files
entropy/client/queryTools.py
(no author) 599e63aca8 - say goodbye to equoTools, say hello to equoInterface
- adapting code to EquoInterface


git-svn-id: http://svn.sabayonlinux.org/projects/entropy/trunk@974 cd1c1023-2f26-0410-ae45-c471fc1f0318
2008-01-02 00:35:02 +00:00

737 lines
26 KiB
Python

#!/usr/bin/python
'''
# DESCRIPTION:
# Entropy database query tools and library
Copyright (C) 2007 Fabio Erculiani
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
'''
import os
from entropyConstants import *
from clientConstants import *
from outputTools import *
from equoInterface import EquoInterface
Equo = EquoInterface()
########################################################
####
## Query Tools
#
def query(options):
rc = 0
if len(options) < 1:
return -10
equoRequestDeep = False
myopts = []
for opt in options:
if (opt == "--deep"):
equoRequestDeep = True
else:
if not opt.startswith("-"):
myopts.append(opt)
if myopts[0] == "installed":
rc = searchInstalledPackages(myopts[1:])
elif myopts[0] == "belongs":
rc = searchBelongs(myopts[1:])
elif myopts[0] == "depends":
rc = searchDepends(myopts[1:])
elif myopts[0] == "files":
rc = searchFiles(myopts[1:])
elif myopts[0] == "needed":
rc = searchNeeded(myopts[1:])
elif myopts[0] == "removal":
rc = searchRemoval(myopts[1:],deep = equoRequestDeep)
elif myopts[0] == "tags":
if (len(myopts) > 1):
rc = searchTaggedPackages(myopts[1:])
elif myopts[0] == "slot":
if (len(myopts) > 1):
rc = searchSlottedPackages(myopts[1:])
elif myopts[0] == "orphans":
rc = searchOrphans()
elif myopts[0] == "list":
mylistopts = options[1:]
if len(mylistopts) > 0:
if mylistopts[0] == "installed":
rc = searchInstalled()
elif myopts[0] == "description":
rc = searchDescription(myopts[1:])
else:
rc = -10
return rc
def searchInstalledPackages(packages, idreturn = False, dbconn = None):
if (not idreturn) and (not etpUi['quiet']):
print_info(brown(" @@ ")+darkgreen("Searching..."))
if not dbconn:
clientDbconn = Equo.clientDbconn
else:
clientDbconn = dbconn
dataInfo = set() # when idreturn is True
for package in packages:
slot = Equo.entropyTools.dep_getslot(package)
tag = Equo.entropyTools.dep_gettag(package)
package = Equo.entropyTools.remove_slot(package)
package = Equo.entropyTools.remove_tag(package)
result = clientDbconn.searchPackages(package, slot = slot, tag = tag)
if (result):
for pkg in result:
idpackage = pkg[1]
if (idreturn):
dataInfo.add(idpackage)
else:
printPackageInfo(idpackage, clientDbconn, clientSearch = True)
# print info
if (not idreturn) and (not etpUi['quiet']):
print_info(blue(" Keyword: ")+bold("\t"+package))
print_info(blue(" Found: ")+bold("\t"+str(len(result)))+red(" entries"))
if (idreturn):
return dataInfo
return 0
def searchBelongs(files, idreturn = False, dbconn = None):
if (not idreturn) and (not etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Belong Search..."))
if not dbconn:
clientDbconn = Equo.clientDbconn
else:
clientDbconn = dbconn
dataInfo = set() # when idreturn is True
results = {}
flatresults = {}
for xfile in files:
like = False
if xfile.find("*") != -1:
xfile.replace("*","%")
like = True
results[xfile] = set()
idpackages = clientDbconn.searchBelongs(xfile, like)
for idpackage in idpackages:
if not flatresults.get(idpackage):
results[xfile].add(idpackage)
flatresults[idpackage] = True
if (results):
for result in results:
# print info
xfile = result
result = results[result]
for idpackage in result:
if (idreturn):
dataInfo.add(idpackage)
elif (etpUi['quiet']):
print clientDbconn.retrieveAtom(idpackage)
else:
printPackageInfo(idpackage, clientDbconn, clientSearch = True)
if (not idreturn) and (not etpUi['quiet']):
print_info(blue(" Keyword: ")+bold("\t"+xfile))
print_info(blue(" Found: ")+bold("\t"+str(len(result)))+red(" entries"))
if (idreturn):
return dataInfo
return 0
def searchDepends(atoms, idreturn = False, dbconn = None):
if (not idreturn) and (not etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Depends Search..."))
if not dbconn:
clientDbconn = Equo.clientDbconn
else:
clientDbconn = dbconn
dataInfo = set() # when idreturn is True
for atom in atoms:
result = clientDbconn.atomMatch(atom)
matchInRepo = False
if (result[0] == -1):
matchInRepo = True
result = Equo.atomMatch(atom)
if (result[0] != -1):
if (matchInRepo):
dbconn = Equo.openRepositoryDatabase(result[1])
else:
dbconn = clientDbconn
searchResults = dbconn.retrieveDepends(result[0])
for idpackage in searchResults:
if (idreturn):
dataInfo.add(idpackage)
else:
if (etpUi['verbose']):
printPackageInfo(idpackage, dbconn, clientSearch = True)
else:
printPackageInfo(idpackage, dbconn, clientSearch = True, strictOutput = True)
# print info
if (not idreturn) and (not etpUi['quiet']):
print_info(blue(" Keyword: ")+bold("\t"+atom))
if (matchInRepo):
where = " from repository "+str(result[1])
else:
where = " from installed packages database"
print_info(blue(" Found: ")+bold("\t"+str(len(searchResults)))+red(" entries")+where)
else:
continue
if (matchInRepo):
dbconn.closeDB()
del dbconn
if (idreturn):
return dataInfo
return 0
def searchNeeded(atoms, idreturn = False, dbconn = None):
if (not idreturn) and (not etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Needed Search..."))
dataInfo = set()
if not dbconn:
clientDbconn = Equo.clientDbconn
else:
clientDbconn = dbconn
for atom in atoms:
match = clientDbconn.atomMatch(atom)
if (match[0] != -1):
# print info
myatom = clientDbconn.retrieveAtom(match[0])
myneeded = clientDbconn.retrieveNeeded(match[0])
for needed in myneeded:
if (idreturn):
dataInfo.add(needed)
elif (etpUi['quiet']):
print needed
else:
print_info(blue(" # ")+red(str(needed)))
if (not idreturn) and (not etpUi['quiet']):
print_info(blue(" Atom: ")+bold("\t"+myatom))
print_info(blue(" Found: ")+bold("\t"+str(len(myneeded)))+red(" libraries"))
if (idreturn):
return dataInfo
return 0
def searchEclass(eclasses, idreturn = False, dbconn = None):
if (not idreturn) and (not etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Eclass Search..."))
if not dbconn:
clientDbconn = Equo.clientDbconn
else:
clientDbconn = dbconn
dataInfo = set()
for eclass in eclasses:
matches = clientDbconn.searchEclassedPackages(eclass, atoms = True)
for match in matches:
if (idreturn):
dataInfo.add(match)
continue
# print info
myatom = match[0]
idpackage = match[1]
if (etpUi['quiet']):
print myatom
else:
if (etpUi['verbose']):
printPackageInfo(idpackage, clientDbconn, clientSearch = True)
else:
printPackageInfo(idpackage, clientDbconn, clientSearch = True, strictOutput = True)
if (not etpUi['quiet']):
print_info(blue(" Found: ")+bold("\t"+str(len(matches)))+red(" packages"))
if (idreturn):
return dataInfo
return 0
def searchFiles(atoms, idreturn = False, dbconn = None):
if (not idreturn) and (not etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Files Search..."))
if not dbconn:
results = searchInstalledPackages(atoms, idreturn = True)
clientDbconn = Equo.clientDbconn
else:
results = searchInstalledPackages(atoms, idreturn = True, dbconn = dbconn)
clientDbconn = dbconn
dataInfo = set() # when idreturn is True
for result in results:
if (result != -1):
files = clientDbconn.retrieveContent(result)
atom = clientDbconn.retrieveAtom(result)
files = list(files)
files.sort()
# print info
if (idreturn):
dataInfo.add((result,files))
else:
if etpUi['quiet']:
for xfile in files:
print xfile
else:
for xfile in files:
print_info(blue(" ### ")+red(xfile))
if (not idreturn) and (not etpUi['quiet']):
print_info(blue(" Package: ")+bold("\t"+atom))
print_info(blue(" Found: ")+bold("\t"+str(len(files)))+red(" files"))
if (idreturn):
return dataInfo
return 0
def searchOrphans():
if (not etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Orphans Search..."))
clientDbconn = Equo.clientDbconn
# start to list all files on the system:
dirs = etpConst['filesystemdirs']
foundFiles = set()
for xdir in dirs:
for currentdir,subdirs,files in os.walk(xdir):
for filename in files:
# filter python compiled objects?
if filename.endswith(".pyo") or filename.startswith(".pyc") or filename == '.keep':
continue
mask = [x for x in etpConst['filesystemdirsmask'] if file.startswith(x)]
if (not mask):
if (not etpUi['quiet']):
print_info(red(" @@ ")+blue("Looking: ")+bold(file[:50]+"..."), back = True)
foundFiles.add(file)
totalfiles = len(foundFiles)
if (not etpUi['quiet']):
print_info(red(" @@ ")+blue("Analyzed directories: ")+' '.join(etpConst['filesystemdirs']))
print_info(red(" @@ ")+blue("Masked directories: ")+' '.join(etpConst['filesystemdirsmask']))
print_info(red(" @@ ")+blue("Number of files collected on the filesystem: ")+bold(str(totalfiles)))
print_info(red(" @@ ")+blue("Now looking into Installed Packages database..."))
# list all idpackages
idpackages = clientDbconn.listAllIdpackages()
# create content list
length = str(len(idpackages))
count = 0
for idpackage in idpackages:
if (not etpUi['quiet']):
count += 1
atom = clientDbconn.retrieveAtom(idpackage)
txt = "["+str(count)+"/"+length+"] "
print_info(red(" @@ ")+blue("Intersecting content of package: ")+txt+bold(atom), back = True)
content = clientDbconn.retrieveContent(idpackage)
_content = set()
for x in content:
if x.startswith("/usr/lib64"):
x = "/usr/lib"+x[len("/usr/lib64"):]
_content.add(x)
# remove from foundFiles
del content
foundFiles.difference_update(_content)
if (not etpUi['quiet']):
print_info(red(" @@ ")+blue("Intersection completed. Showing statistics: "))
print_info(red(" @@ ")+blue("Number of total files: ")+bold(str(totalfiles)))
print_info(red(" @@ ")+blue("Number of matching files: ")+bold(str(totalfiles - len(foundFiles))))
print_info(red(" @@ ")+blue("Number of orphaned files: ")+bold(str(len(foundFiles))))
# order
foundFiles = list(foundFiles)
foundFiles.sort()
if (not etpUi['quiet']):
print_info(red(" @@ ")+blue("Writing file to disk: ")+bold("/tmp/equo-orphans.txt"))
f = open("/tmp/equo-orphans.txt","w")
for x in foundFiles:
f.write(x+"\n")
f.flush()
f.close()
return 0
else:
for x in foundFiles:
print x
return 0
def searchRemoval(atoms, idreturn = False, deep = False):
clientDbconn = Equo.clientDbconn
if (not idreturn) and (not etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Removal Search..."))
foundAtoms = []
for atom in atoms:
match = clientDbconn.atomMatch(atom)
if match[1] == 0:
foundAtoms.append(match[0])
# are packages in foundAtoms?
if (len(foundAtoms) == 0):
print_error(red("No packages found."))
return 127,-1
choosenRemovalQueue = []
if (not etpUi['quiet']):
print_info(red(" @@ ")+blue("Calculating removal dependencies, please wait..."), back = True)
treeview = Equo.generate_depends_tree(foundAtoms, deep = deep)
treelength = len(treeview[0])
if treelength > 1:
treeview = treeview[0]
for x in range(treelength)[::-1]:
for y in treeview[x]:
choosenRemovalQueue.append(y)
if (choosenRemovalQueue):
if (not etpUi['quiet']):
print_info(red(" @@ ")+blue("These are the packages that would added to the removal queue:"))
totalatoms = str(len(choosenRemovalQueue))
atomscounter = 0
for idpackage in choosenRemovalQueue:
atomscounter += 1
rematom = clientDbconn.retrieveAtom(idpackage)
if (not etpUi['quiet']):
installedfrom = clientDbconn.retrievePackageFromInstalledTable(idpackage)
repositoryInfo = bold("[")+red("from: ")+brown(installedfrom)+bold("]")
stratomscounter = str(atomscounter)
while len(stratomscounter) < len(totalatoms):
stratomscounter = " "+stratomscounter
print_info(" # "+red("(")+bold(stratomscounter)+"/"+blue(str(totalatoms))+red(")")+repositoryInfo+" "+blue(rematom))
else:
print rematom
if (idreturn):
return treeview
return 0
def searchInstalled(idreturn = False):
if (not idreturn) and (not etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Installed Search..."))
clientDbconn = Equo.clientDbconn
installedPackages = clientDbconn.listAllPackages()
installedPackages.sort()
if not idreturn:
if (not etpUi['quiet']):
print_info(red(" @@ ")+blue("These are the installed packages:"))
for package in installedPackages:
if (not etpUi['verbose']):
atom = Equo.entropyTools.dep_getkey(package[0])
else:
atom = package[0]
branchinfo = ""
if (etpUi['verbose']):
branchinfo = darkgreen(" [")+red(package[2])+darkgreen("]")
if (not etpUi['quiet']):
print_info(red(" #")+blue(str(package[1]))+branchinfo+" "+atom)
else:
print atom
return 0
else:
idpackages = set()
for x in installedPackages:
idpackages.add(x[1])
return list(idpackages)
def searchPackage(packages, idreturn = False):
foundPackages = {}
dataInfo = set() # when idreturn is True
if (not idreturn) or (etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Searching..."))
# search inside each available database
repoNumber = 0
for repo in etpRepositories:
foundPackages[repo] = {}
repoNumber += 1
if (not idreturn) and (not etpUi['quiet']):
print_info(blue(" #"+str(repoNumber))+bold(" "+etpRepositories[repo]['description']))
dbconn = Equo.openRepositoryDatabase(repo)
for package in packages:
slot = Equo.entropyTools.dep_getslot(package)
tag = Equo.entropyTools.dep_gettag(package)
package = Equo.entropyTools.remove_slot(package)
package = Equo.entropyTools.remove_tag(package)
result = dbconn.searchPackages(package, slot = slot, tag = tag)
if (not result): # look for provide
result = dbconn.searchProvide(package, slot = slot, tag = tag)
if (result):
foundPackages[repo][package] = result
# print info
for pkg in foundPackages[repo][package]:
idpackage = pkg[1]
if (idreturn):
dataInfo.add((idpackage,repo))
else:
printPackageInfo(idpackage,dbconn)
if (not idreturn) and (not etpUi['quiet']):
print_info(blue(" Keyword: ")+bold("\t"+package))
print_info(blue(" Found: ")+bold("\t"+str(len(foundPackages[repo][package])))+red(" entries"))
if (idreturn):
return dataInfo
return 0
def searchSlottedPackages(slots, datareturn = False, dbconn = None):
foundPackages = {}
dbclose = True
if dbconn:
dbclose = False
if (not datareturn) and (not etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Slot Search..."))
# search inside each available database
repoNumber = 0
for repo in etpRepositories:
foundPackages[repo] = {}
repoNumber += 1
if (not datareturn) and (not etpUi['quiet']):
print_info(blue(" #"+str(repoNumber))+bold(" "+etpRepositories[repo]['description']))
if dbclose:
dbconn = Equo.openRepositoryDatabase(repo)
for slot in slots:
results = dbconn.searchSlottedPackages(slot, atoms = True)
for result in results:
foundPackages[repo][result[1]] = result[0]
# print info
if (not datareturn):
printPackageInfo(result[1],dbconn)
if (not datareturn) and (not etpUi['quiet']):
print_info(blue(" Keyword: ")+bold("\t"+slot))
print_info(blue(" Found: ")+bold("\t"+str(len(results)))+red(" entries"))
if (datareturn):
return foundPackages
return 0
def searchTaggedPackages(tags, datareturn = False, dbconn = None):
foundPackages = {}
dbclose = True
if dbconn:
dbclose = False
if (not datareturn) and (not etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Tag Search..."))
# search inside each available database
repoNumber = 0
for repo in etpRepositories:
foundPackages[repo] = {}
repoNumber += 1
if (not datareturn) and (not etpUi['quiet']):
print_info(blue(" #"+str(repoNumber))+bold(" "+etpRepositories[repo]['description']))
if dbclose:
dbconn = Equo.openRepositoryDatabase(repo)
for tag in tags:
results = dbconn.searchTaggedPackages(tag, atoms = True)
for result in results:
foundPackages[repo][result[1]] = result[0]
# print info
if (not datareturn):
printPackageInfo(result[1],dbconn)
if (not datareturn) and (not etpUi['quiet']):
print_info(blue(" Keyword: ")+bold("\t"+tag))
print_info(blue(" Found: ")+bold("\t"+str(len(results)))+red(" entries"))
if (datareturn):
return foundPackages
return 0
def searchDescription(descriptions, idreturn = False):
foundPackages = {}
if (not idreturn) and (not etpUi['quiet']):
print_info(darkred(" @@ ")+darkgreen("Description Search..."))
# search inside each available database
repoNumber = 0
for repo in etpRepositories:
foundPackages[repo] = {}
repoNumber += 1
if (not idreturn) and (not etpUi['quiet']):
print_info(blue(" #"+str(repoNumber))+bold(" "+etpRepositories[repo]['description']))
dbconn = Equo.openRepositoryDatabase(repo)
dataInfo, descdata = __searchDescriptions(descriptions, dbconn, idreturn)
foundPackages[repo].update(descdata)
if (idreturn):
return dataInfo
return 0
'''
Internal functions
'''
def __searchDescriptions(descriptions, dbconn, idreturn = False):
dataInfo = set() # when idreturn is True
mydescdata = {}
for desc in descriptions:
result = dbconn.searchPackagesByDescription(desc)
if (result):
mydescdata[desc] = result
for pkg in mydescdata[desc]:
idpackage = pkg[1]
if (idreturn):
dataInfo.add(idpackage)
elif (etpUi['quiet']):
print dbconn.retrieveAtom(idpackage)
else:
printPackageInfo(idpackage,dbconn)
# print info
if (not idreturn) and (not etpUi['quiet']):
print_info(blue(" Keyword: ")+bold("\t"+desc))
print_info(blue(" Found: ")+bold("\t"+str(len(mydescdata[desc])))+red(" entries"))
return dataInfo,mydescdata
def printPackageInfo(idpackage, dbconn, clientSearch = False, strictOutput = False, extended = False):
# now fetch essential info
pkgatom = dbconn.retrieveAtom(idpackage)
if (etpUi['quiet']):
print pkgatom
return
if (not strictOutput):
pkgname = dbconn.retrieveName(idpackage)
pkgcat = dbconn.retrieveCategory(idpackage)
pkglic = dbconn.retrieveLicense(idpackage)
pkgsize = dbconn.retrieveSize(idpackage)
pkgbin = dbconn.retrieveDownloadURL(idpackage)
pkgflags = dbconn.retrieveCompileFlags(idpackage)
pkgkeywords = dbconn.retrieveKeywords(idpackage)
pkgdigest = dbconn.retrieveDigest(idpackage)
pkgcreatedate = Equo.entropyTools.convertUnixTimeToHumanTime(float(dbconn.retrieveDateCreation(idpackage)))
pkgsize = Equo.entropyTools.bytesIntoHuman(pkgsize)
pkgdeps = dbconn.retrieveDependencies(idpackage)
pkgconflicts = dbconn.retrieveConflicts(idpackage)
pkghome = dbconn.retrieveHomepage(idpackage)
pkgslot = dbconn.retrieveSlot(idpackage)
pkgver = dbconn.retrieveVersion(idpackage)
pkgtag = dbconn.retrieveVersionTag(idpackage)
pkgrev = dbconn.retrieveRevision(idpackage)
pkgdesc = dbconn.retrieveDescription(idpackage)
pkgbranch = dbconn.retrieveBranch(idpackage)
if (not pkgtag):
pkgtag = "NoTag"
if (not clientSearch):
# client info
installedVer = "Not installed"
installedTag = "N/A"
installedRev = "N/A"
pkginstalled = Equo.clientDbconn.atomMatch(Equo.entropyTools.dep_getkey(pkgatom), matchSlot = pkgslot)
if (pkginstalled[1] == 0):
idx = pkginstalled[0]
# found
installedVer = Equo.clientDbconn.retrieveVersion(idx)
installedTag = Equo.clientDbconn.retrieveVersionTag(idx)
if not installedTag:
installedTag = "NoTag"
installedRev = Equo.clientDbconn.retrieveRevision(idx)
print_info(red(" @@ Package: ")+bold(pkgatom)+"\t\t"+blue("branch: ")+bold(pkgbranch))
if (not strictOutput):
print_info(darkgreen(" Category:\t\t")+blue(pkgcat))
print_info(darkgreen(" Name:\t\t\t")+blue(pkgname))
print_info(darkgreen(" Available:\t\t")+blue("version: ")+bold(pkgver)+blue(" ~ tag: ")+bold(pkgtag)+blue(" ~ revision: ")+bold(str(pkgrev)))
if (not clientSearch):
print_info(darkgreen(" Installed:\t\t")+blue("version: ")+bold(installedVer)+blue(" ~ tag: ")+bold(installedTag)+blue(" ~ revision: ")+bold(str(installedRev)))
if (not strictOutput):
print_info(darkgreen(" Slot:\t\t\t")+blue(str(pkgslot)))
print_info(darkgreen(" Size:\t\t\t")+blue(str(pkgsize)))
print_info(darkgreen(" Download:\t\t")+brown(str(pkgbin)))
print_info(darkgreen(" Checksum:\t\t")+brown(str(pkgdigest)))
if (pkgdeps):
print_info(darkred(" ##")+darkgreen(" Dependencies:"))
for pdep in pkgdeps:
print_info(darkred(" ## \t\t\t")+brown(pdep))
if (pkgconflicts):
print_info(darkred(" ##")+darkgreen(" Conflicts:"))
for conflict in pkgconflicts:
print_info(darkred(" ## \t\t\t")+brown(conflict))
print_info(darkgreen(" Homepage:\t\t")+red(pkghome))
print_info(darkgreen(" Description:\t\t")+pkgdesc)
if (not strictOutput):
if (extended):
print_info(darkgreen(" CHOST:\t\t")+blue(pkgflags[0]))
print_info(darkgreen(" CFLAGS:\t\t")+red(pkgflags[1]))
print_info(darkgreen(" CXXFLAGS:\t\t")+blue(pkgflags[2]))
sources = dbconn.retrieveSources(idpackage)
eclasses = dbconn.retrieveEclasses(idpackage)
etpapi = dbconn.retrieveApi(idpackage)
print_info(darkgreen(" Gentoo eclasses:\t")+red(' '.join(eclasses)))
if (sources):
print_info(darkgreen(" Sources:"))
for source in sources:
print_info(darkred(" # Source: ")+blue(source))
print_info(darkgreen(" Entry API:\t\t")+red(str(etpapi)))
else:
print_info(darkgreen(" Compiled with:\t")+blue(pkgflags[1]))
print_info(darkgreen(" Keywords:\t\t")+red(' '.join(pkgkeywords)))
print_info(darkgreen(" Created:\t\t")+pkgcreatedate)
print_info(darkgreen(" License:\t\t")+red(pkglic))