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() { async parent() {
await this.classify() await this.classify()
if ( this.is_directory() ) return path.resolve(this.path, '..') if ( this.is_directory() ) return path.join(this.path, '..')
return this.directory() 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() { name() {
return path.basename(this.path) 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 ImplementationError = require('libflitter/errors/ImplementationError')
const PermissionDeniedError = require('../logical/error/PermissionDeniedError') const PermissionDeniedError = require('../logical/error/PermissionDeniedError')
const CommandNotFoundError = require('../logical/error/CommandNotFoundError') const CommandNotFoundError = require('../logical/error/CommandNotFoundError')
const FilesystemResourceNotFoundError = require('../logical/error/FilesystemResourceNotFoundError')
const uuid = require('uuid/v4') const uuid = require('uuid/v4')
const UniversalPath = require('../logical/UniversalPath') const UniversalPath = require('../logical/UniversalPath')
const SystemMetrics = require('../logical/SystemMetrics') const SystemMetrics = require('../logical/SystemMetrics')
@ -181,7 +182,7 @@ class Host extends Injectable {
const result = await this.execute(command) const result = await this.execute(command)
if ( result.exit_code !== 0 || result.clean_out.length < 1 ) { if ( result.exit_code !== 0 || result.clean_out.length < 1 ) {
const E = this._get_result_error_class(result) 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()) return this.utility.infer(result.clean_out[0].trim())
} }
@ -190,7 +191,7 @@ class Host extends Injectable {
const result = await this.execute(command) const result = await this.execute(command)
if ( result.exit_code !== 0 ) { if ( result.exit_code !== 0 ) {
const E = this._get_result_error_class(result) 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 return result
} }
@ -218,6 +219,13 @@ class Host extends Injectable {
'command not found', '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 ) { for ( const phrase of access_denied_phrases ) {
if ( both.includes(phrase) ) return PermissionDeniedError if ( both.includes(phrase) ) return PermissionDeniedError
} }
@ -226,6 +234,10 @@ class Host extends Injectable {
if ( both.includes(phrase) ) return CommandNotFoundError if ( both.includes(phrase) ) return CommandNotFoundError
} }
for ( const phrase of resource_not_found ) {
if ( both.includes(phrase) ) return FilesystemResourceNotFoundError
}
return Error return Error
} }
} }