Improve cmd error classification; add Git repository support

This commit is contained in:
garrettmills 2020-03-05 10:25:45 -06:00
parent ec8047361c
commit 24c18e4277
No known key found for this signature in database
GPG Key ID: 6ACD58D6ADACFC6E
4 changed files with 102 additions and 3 deletions

View File

@ -0,0 +1,76 @@
const { Injectable } = require('flitter-di')
const UniversalPath = require('../logical/UniversalPath')
class Repository extends Injectable {
static fromString(string) {
const path = UniversalPath.fromString(string)
return new this(path)
}
constructor(root_path) {
super()
this._host = root_path.host
this._path = root_path
}
async is_repo() {
const git_dir = await this._path.child('.git')
await git_dir.classify()
return git_dir.is_directory()
}
async init() {
if ( !(await this.is_repo()) ) {
await this._path.classify()
if ( !this._path.is_valid() ) await this._path.touch(true)
await this._git_cmd(`init`)
}
}
async destroy() {
const git_dir = await this._path.child('.git')
await git_dir.unlink()
}
async clone(from) {
await this._path.classify()
if ( !this._path.is_valid() ) await this._path.touch(true)
await this._git_cmd(`clone ${from} ${this._path.path}`)
}
async stage(...blobs) {
await this._git_cmd(`add ${blobs.map(x => '"'+x+'"').join(' ')}`)
}
async unstage(...blobs) {
await this._git_cmd(`reset ${blobs.map(x => '"'+x+'"').join(' ')}`)
}
async commit(message) {
await this._git_cmd(`commit -m "${message}"`)
}
async tag(label) {
await this._git_cmd(`tag "${label}"`)
}
async push(target = false, ref = false) {
await this._git_cmd(`push${target ? ' '+target : ''}${ref ? ' '+ref : ''} --tags`)
}
async checkout(ref = 'master') {
await this._git_cmd(`checkout "${ref}"`)
}
async stash() {
await this._git_cmd(`stash`)
}
async _git_cmd(command) {
const cd_cmd = await this._host.get_directory_change_command(this._path)
const cmd = `${cd_cmd} && git ${command}`
return this._host.run(cmd)
}
}
module.exports = exports = Repository

View File

@ -65,10 +65,16 @@ class UniversalPath extends Injectable {
async parent() {
await this.classify()
if ( this.is_directory() ) return path.resolve(this.path, '..')
if ( this.is_directory() ) return path.join(this.path, '..')
return this.directory()
}
async child(subpath) {
await this.classify()
if ( !this.is_directory() ) throw new Error('Cannot get sub-path: not a directory: '+this.path)
return this.constructor.fromString(path.join(String(this), subpath))
}
name() {
return path.basename(this.path)
}

View File

@ -0,0 +1,5 @@
class FilesystemResourceNotFoundError extends Error {
}
module.exports = exports = FilesystemResourceNotFoundError

View File

@ -2,6 +2,7 @@ const { Injectable } = require('flitter-di')
const ImplementationError = require('libflitter/errors/ImplementationError')
const PermissionDeniedError = require('../logical/error/PermissionDeniedError')
const CommandNotFoundError = require('../logical/error/CommandNotFoundError')
const FilesystemResourceNotFoundError = require('../logical/error/FilesystemResourceNotFoundError')
const uuid = require('uuid/v4')
const UniversalPath = require('../logical/UniversalPath')
const SystemMetrics = require('../logical/SystemMetrics')
@ -181,7 +182,7 @@ class Host extends Injectable {
const result = await this.execute(command)
if ( result.exit_code !== 0 || result.clean_out.length < 1 ) {
const E = this._get_result_error_class(result)
throw new E('Unable to get line output from command: '+command)
throw new E('Unable to get line output from command: '+command+'\n'+result.clean_err)
}
return this.utility.infer(result.clean_out[0].trim())
}
@ -190,7 +191,7 @@ class Host extends Injectable {
const result = await this.execute(command)
if ( result.exit_code !== 0 ) {
const E = this._get_result_error_class(result)
throw new E('Unable to run command: '+command)
throw new E('Unable to run command: '+command+'\n'+result.clean_err)
}
return result
}
@ -218,6 +219,13 @@ class Host extends Injectable {
'command not found',
]
const resource_not_found = [
'no such file',
'no such directory',
'no such file or directory',
'no such directory or file',
]
for ( const phrase of access_denied_phrases ) {
if ( both.includes(phrase) ) return PermissionDeniedError
}
@ -226,6 +234,10 @@ class Host extends Injectable {
if ( both.includes(phrase) ) return CommandNotFoundError
}
for ( const phrase of resource_not_found ) {
if ( both.includes(phrase) ) return FilesystemResourceNotFoundError
}
return Error
}
}