src/nifty.nim
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 |
import
json,
os,
parseopt,
logging,
algorithm,
strutils,
sequtils
import
niftypkg/niftylogger
newNiftyLogger().addHandler()
setLogFilter(lvlInfo)
import
niftypkg/config,
niftypkg/project,
niftypkg/messaging
let usage* = """ $1 v$2 - $3
(c) 2017-2020 $4
Usage:
nifty <command> [<package>] Executes <command> (on <package>).
=> For more information on available commands, run: nifty help
Options:
--log, -l Specifies the log level (debug|info|notice|warn|error|fatal).
Default: info
--force, -f Do not ask for confirmation when executing the specified command.
--help, -h Displays this message.
--version, -h Displays the version of the application.
""" % [pkgTitle, pkgVersion, pkgDescription, pkgAuthor]
var force = false
# Helper Methods
proc addProperty(parentObj: JsonNode, name = ""): tuple[key: string, value: JsonNode] =
var done = false
while (not done):
if name == "":
result.key = editValue("Name")
elif name == "name":
warn "Property identifier 'name' cannot be modified."
else:
printValue(" Name", name)
result.key = name
var ok = false
while (not ok):
var value = ""
if parentObj.hasKey(result.key):
value = $parentObj[result.key]
try:
result.value = editValue("Value", value).parseJson
if (result.value == newJNull()):
ok = confirm("Remove property '$1'?" % result.key)
done = true
else:
ok = true
except:
warn("Please enter a valid JSON value.")
done = done or confirm("OK?")
proc addProperties(obj: var JsonNode) =
var done = false
while (not done):
let prop = addProperty(obj)
obj[prop.key] = prop.value
done = not confirm("Do you want to add/remove more properties?")
proc changeValue(oldv: tuple[label: string, value: JsonNode], newv: tuple[label: string, value: JsonNode]): bool =
if oldv.value != newJNull():
printDeleted(oldv.label, $oldv.value)
if newv.value != newJNull():
printAdded(newv.label, $newv.value)
return confirm("Confirm change?")
proc confirmAndRemoveDir(dir: string) =
let answer = force or confirm "Delete directory '$1' and all its contents?" % dir
if answer:
dir.removeDir()
proc confirmAndRemoveFile(file: string) =
let answer = force or confirm "Delete file '$1'? [y/n]" % file
if answer:
file.removeFile()
proc confirmAndRemovePackage(pkg: string) =
if pkg.fileExists():
pkg.confirmAndRemoveFile()
elif pkg.dirExists():
pkg.confirmAndRemoveDir()
else:
warn "Package '$1' not found." % pkg
proc buildTree(prj: NiftyProject, dir: string): TreeNode =
var node = newTreeNode(dir.extractFilename)
for k, v in prj.packages.pairs:
var d = dir / prj.storage / k
var p = newNiftyProject(d)
if p.configured:
p.load
node.add buildTree(p, d)
else:
node.add newTreeNode(k)
return node
proc updateDefinitions(prj: var NiftyProject): bool =
result = false
let sysCommands = niftyTpl.parseJson["commands"]
for k, v in sysCommands.pairs:
if prj.commands.hasKey(k):
let sysCommand = sysCommands[k]
var prjCommand = prj.commands[k]
for prop, val in sysCommand.pairs:
let sysProp = sysCommand[prop]
var prjProp = newJNull()
if prjCommand.hasKey(prop):
prjProp = prjCommand[prop]
if prjProp != newJNull():
if prjProp != sysProp:
let sysVal = (label: k & "." & prop, value: sysProp)
let prjVal = (label: k & "." & prop, value: prjProp)
if changeValue(prjVal, sysVal):
prjCommand[prop] = sysProp
result = true
else:
result = true
# Adding new property
printAdded("$1.$2" % [k, prop], $sysProp)
prjCommand[prop] = sysProp
else:
result = true
# Adding new command
printAdded(k, $sysCommands[k])
prj.commands[k] = sysCommands[k]
### MAIN ###
var args = newSeq[string](0)
for kind, key, val in getopt():
case kind:
of cmdArgument:
args.add key
of cmdLongOption, cmdShortOption:
case key:
of "force", "f":
force = true
of "log", "l":
var val = val
setLogLevel(val)
of "help", "h":
echo usage
quit(0)
of "version", "v":
echo pkgVersion
quit(0)
else:
discard
else:
discard
var prj = newNiftyProject(getCurrentDir())
if args.len == 0:
echo usage
quit(0)
case args[0]:
of "init":
if prj.configured:
fatal "Project already configured."
quit(2)
var storage = "packages"
if args.len > 2:
storage = args[1]
prj.init(storage)
notice "Project initialized using '$1' as storage directory." % storage
of "purge":
prj.load
confirmAndRemoveDir(prj.dir/prj.storage)
of "map":
if args.len < 2:
fatal "No package specified."
quit(3)
let alias = args[1]
var props = newJObject()
prj.load
if prj.packages.hasKey(alias):
notice "Remapping existing package: " & alias
warn "Specify properties for package '$1':" % alias
props = prj.packages[alias]
for k, v in props.mpairs:
if k == "name":
continue
let prop = addProperty(props, k)
props[prop.key] = prop.value
if confirm "Do you want to add/remove more properties?":
addProperties(props)
else:
notice "Mapping new package: " & alias
warn "Specify properties for package '$1':" % alias
addProperties(props)
prj.map(alias, props)
of "unmap":
if args.len < 2:
fatal "No package specified."
quit(3)
let alias = args[1]
prj.load
if not prj.packages.hasKey(alias):
fatal "Package '$1' not defined." % [alias]
quit(4)
if force or confirm("Remove mapping for package '$1'?" % alias):
prj.unmap(alias)
of "remove":
prj.load
if args.len < 2:
var packages = toSeq(prj.packages.pairs)
if packages.len == 0:
warn "No packages defined - nothing to do."
else:
for key, val in prj.packages.pairs:
confirmAndRemovePackage(prj.storage/key)
else:
confirmAndRemovePackage(prj.storage/args[1])
of "list":
prj.load
let pwd = getCurrentDir()
echo buildTree(prj, pwd).tree
of "info":
if args.len < 2:
fatal "No package specified."
quit(3)
prj.load
let alias = args[1]
if not prj.packages.hasKey(alias):
fatal "Package '$1' not defined." % [alias]
quit(4)
let data = prj.packages[alias]
for k, v in data.pairs:
echo "$1:\t$2" % [k, $v]
of "help":
echo ""
if args.len < 2:
var sortedKeys = toSeq(prj.help.keys)
sortedKeys.sort(cmp[string])
for k in sortedKeys:
printGreen " nifty $1" % prj.help[k]["_syntax"].getStr
echo "\n $1\n" % prj.help[k]["_description"].getStr
else:
let cmd = args[1]
let help = prj.help[cmd]
if not prj.help.hasKey(cmd):
fatal "Command '$1' is not defined." % cmd
quit(5)
printGreen " nifty " & help["_syntax"].getStr
echo "\n $1\n" % help["_description"].getStr
of "update":
prj.load
if updateDefinitions(prj):
prj.save
else:
if args.len < 1:
echo usage
quit(1)
if args.len < 2:
prj.load
var packages = toSeq(prj.packages.pairs)
if packages.len == 0:
warn "No packages defined - nothing to do."
quit(0)
if args[0].startsWith("$"):
# Execute task list
let tasklist = args[0][1.. args[0].len - 1]
if prj.tasklists.isNil or not prj.tasklists.hasKey(tasklist):
warn "Task list '$1' not defined in project" % [tasklist]
quit(0)
let tasks = prj.tasklists[tasklist].getElems
for task in tasks:
let targs = task.getStr.split(" ").mapIt(it.strip)
discard execute(prj, targs[0], targs[1])
quit(0)
for key, val in prj.packages.pairs:
prj.executeRec(args[0], key)
else:
prj.executeRec(args[0], args[1])
|