2019-03-04 09:28:35 +01:00
|
|
|
if exists('g:polyglot_disabled') && index(g:polyglot_disabled, 'crystal') != -1
|
|
|
|
finish
|
|
|
|
endif
|
|
|
|
|
2018-01-10 23:50:02 +01:00
|
|
|
" ___vital___
|
|
|
|
" NOTE: lines between '" ___vital___' is generated by :Vitalize.
|
|
|
|
" Do not mofidify the code nor insert new lines before '" ___vital___'
|
|
|
|
function! s:_SID() abort
|
|
|
|
return matchstr(expand('<sfile>'), '<SNR>\zs\d\+\ze__SID$')
|
|
|
|
endfunction
|
|
|
|
execute join(['function! vital#_crystal#Process#import() abort', printf("return map({'shellescape': '', 'has_vimproc': '', 'system': '', 'iconv': '', 'spawn': '', 'get_last_status': ''}, \"vital#_crystal#function('<SNR>%s_' . v:key)\")", s:_SID()), 'endfunction'], "\n")
|
|
|
|
delfunction s:_SID
|
|
|
|
" ___vital___
|
2016-05-02 10:49:45 +02:00
|
|
|
" TODO: move all comments to doc file.
|
|
|
|
"
|
|
|
|
"
|
|
|
|
" FIXME: This module name should be Vital.System ?
|
|
|
|
" But the name has been already taken.
|
|
|
|
|
|
|
|
let s:save_cpo = &cpo
|
|
|
|
set cpo&vim
|
|
|
|
|
|
|
|
|
|
|
|
" FIXME: Unfortunately, can't use s:_vital_loaded() for this purpose.
|
|
|
|
" Because these variables are used when this script file is loaded.
|
|
|
|
let s:is_windows = has('win16') || has('win32') || has('win64') || has('win95')
|
|
|
|
let s:is_unix = has('unix')
|
|
|
|
" As of 7.4.122, the system()'s 1st argument is converted internally by Vim.
|
|
|
|
" Note that Patch 7.4.122 does not convert system()'s 2nd argument and
|
|
|
|
" return-value. We must convert them manually.
|
|
|
|
let s:need_trans = v:version < 704 || (v:version == 704 && !has('patch122'))
|
|
|
|
|
|
|
|
let s:TYPE_DICT = type({})
|
|
|
|
let s:TYPE_LIST = type([])
|
2018-01-10 23:50:02 +01:00
|
|
|
let s:TYPE_STRING = type('')
|
2016-05-02 10:49:45 +02:00
|
|
|
|
|
|
|
function! s:spawn(expr, ...) abort
|
2018-01-10 23:50:02 +01:00
|
|
|
if type(a:expr) is s:TYPE_LIST
|
|
|
|
let special = 1
|
|
|
|
let cmdline = join(map(a:expr, 's:shellescape(v:val, special)'), ' ')
|
|
|
|
elseif type(a:expr) is s:TYPE_STRING
|
|
|
|
let cmdline = a:expr
|
|
|
|
if a:0 && a:1
|
|
|
|
" for :! command
|
|
|
|
let cmdline = substitute(cmdline, '\([!%#]\|<[^<>]\+>\)', '\\\1', 'g')
|
|
|
|
endif
|
|
|
|
else
|
|
|
|
throw 'vital: Process: invalid argument (value type:' . type(a:expr) . ')'
|
|
|
|
endif
|
2016-05-02 10:49:45 +02:00
|
|
|
if s:is_windows
|
2018-01-10 23:50:02 +01:00
|
|
|
silent execute '!start' cmdline
|
|
|
|
else
|
|
|
|
silent execute '!' cmdline '&'
|
2016-05-02 10:49:45 +02:00
|
|
|
endif
|
|
|
|
return ''
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
" iconv() wrapper for safety.
|
|
|
|
function! s:iconv(expr, from, to) abort
|
2018-01-10 23:50:02 +01:00
|
|
|
if a:from ==# '' || a:to ==# '' || a:from ==? a:to
|
2016-05-02 10:49:45 +02:00
|
|
|
return a:expr
|
|
|
|
endif
|
|
|
|
let result = iconv(a:expr, a:from, a:to)
|
2018-01-10 23:50:02 +01:00
|
|
|
return result !=# '' ? result : a:expr
|
2016-05-02 10:49:45 +02:00
|
|
|
endfunction
|
|
|
|
|
|
|
|
" Check vimproc.
|
|
|
|
function! s:has_vimproc() abort
|
|
|
|
if !exists('s:exists_vimproc')
|
|
|
|
try
|
|
|
|
call vimproc#version()
|
|
|
|
let s:exists_vimproc = 1
|
|
|
|
catch
|
|
|
|
let s:exists_vimproc = 0
|
|
|
|
endtry
|
|
|
|
endif
|
|
|
|
return s:exists_vimproc
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
" * {command} [, {input} [, {timeout}]]
|
|
|
|
" * {command} [, {dict}]
|
|
|
|
" {dict} = {
|
|
|
|
" use_vimproc: bool,
|
|
|
|
" input: string,
|
|
|
|
" timeout: bool,
|
|
|
|
" background: bool,
|
|
|
|
" }
|
|
|
|
function! s:system(str, ...) abort
|
|
|
|
" Process optional arguments at first
|
|
|
|
" because use_vimproc is required later
|
|
|
|
" for a:str argument.
|
|
|
|
let input = ''
|
|
|
|
let use_vimproc = s:has_vimproc()
|
|
|
|
let background = 0
|
|
|
|
let args = []
|
|
|
|
if a:0 ==# 1
|
|
|
|
" {command} [, {dict}]
|
|
|
|
" a:1 = {dict}
|
|
|
|
if type(a:1) is s:TYPE_DICT
|
|
|
|
if has_key(a:1, 'use_vimproc')
|
|
|
|
let use_vimproc = a:1.use_vimproc
|
|
|
|
endif
|
|
|
|
if has_key(a:1, 'input')
|
|
|
|
let args += [s:iconv(a:1.input, &encoding, 'char')]
|
|
|
|
endif
|
|
|
|
if use_vimproc && has_key(a:1, 'timeout')
|
|
|
|
" ignores timeout unless you have vimproc.
|
|
|
|
let args += [a:1.timeout]
|
|
|
|
endif
|
|
|
|
if has_key(a:1, 'background')
|
|
|
|
let background = a:1.background
|
|
|
|
endif
|
|
|
|
elseif type(a:1) is s:TYPE_STRING
|
|
|
|
let args += [s:iconv(a:1, &encoding, 'char')]
|
|
|
|
else
|
2018-01-10 23:50:02 +01:00
|
|
|
throw 'vital: Process: invalid argument (value type:' . type(a:1) . ')'
|
2016-05-02 10:49:45 +02:00
|
|
|
endif
|
|
|
|
elseif a:0 >= 2
|
|
|
|
" {command} [, {input} [, {timeout}]]
|
|
|
|
" a:000 = [{input} [, {timeout}]]
|
|
|
|
let [input; rest] = a:000
|
|
|
|
let input = s:iconv(input, &encoding, 'char')
|
|
|
|
let args += [input] + rest
|
|
|
|
endif
|
|
|
|
|
|
|
|
" Process a:str argument.
|
|
|
|
if type(a:str) is s:TYPE_LIST
|
|
|
|
let expr = use_vimproc ? '"''" . v:val . "''"' : 's:shellescape(v:val)'
|
|
|
|
let command = join(map(copy(a:str), expr), ' ')
|
|
|
|
elseif type(a:str) is s:TYPE_STRING
|
|
|
|
let command = a:str
|
|
|
|
else
|
2018-01-10 23:50:02 +01:00
|
|
|
throw 'vital: Process: invalid argument (value type:' . type(a:str) . ')'
|
2016-05-02 10:49:45 +02:00
|
|
|
endif
|
|
|
|
if s:need_trans
|
|
|
|
let command = s:iconv(command, &encoding, 'char')
|
|
|
|
endif
|
|
|
|
let args = [command] + args
|
|
|
|
if background && (use_vimproc || !s:is_windows)
|
2018-01-10 23:50:02 +01:00
|
|
|
if has('nvim')
|
|
|
|
throw "vital: Process: neovim's system() doesn't support background(&) process (cmdline:" . string(a:str) . ')'
|
|
|
|
endif
|
2016-05-02 10:49:45 +02:00
|
|
|
let args[0] = args[0] . ' &'
|
|
|
|
endif
|
|
|
|
|
|
|
|
let funcname = use_vimproc ? 'vimproc#system' : 'system'
|
|
|
|
let output = call(funcname, args)
|
|
|
|
let output = s:iconv(output, 'char', &encoding)
|
|
|
|
return output
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
function! s:get_last_status() abort
|
|
|
|
return s:has_vimproc() ?
|
|
|
|
\ vimproc#get_last_status() : v:shell_error
|
|
|
|
endfunction
|
|
|
|
|
|
|
|
if s:is_windows
|
2018-01-10 23:50:02 +01:00
|
|
|
function! s:shellescape(...) abort
|
|
|
|
try
|
|
|
|
let shellslash = &shellslash
|
|
|
|
set noshellslash
|
|
|
|
return call('shellescape', a:000)
|
|
|
|
finally
|
|
|
|
let &shellslash = shellslash
|
|
|
|
endtry
|
2016-05-02 10:49:45 +02:00
|
|
|
endfunction
|
|
|
|
else
|
|
|
|
function! s:shellescape(...) abort
|
|
|
|
return call('shellescape', a:000)
|
|
|
|
endfunction
|
|
|
|
endif
|
|
|
|
|
|
|
|
|
|
|
|
let &cpo = s:save_cpo
|
|
|
|
unlet s:save_cpo
|
|
|
|
|
|
|
|
" vim:set et ts=2 sts=2 sw=2 tw=0:
|