1 (изменено: Rumata, 2016-07-21 19:16:32)

Тема: CMD/BAT + JS/VBS/PS1/WSF/HTA/PL/PY/BASH: Два в одном

Компиляция известных решений:
CMD/BAT + JavaScript: два в одном
JScript:CMD - вариант с условной компиляцией

CMD/BAT + VBScript: два в одном
CMD/BAT + HTML: два в одном

А также:
PowerShell в BAT на StackOverflow
WSF в BAT (забугорный форум, специализирующийся на пакетных файлах; в теме перекрестные ссылки с нашим форумом)

Предлагаемый скрипт cmdize.bat конвертирует оригинальный файл (JScript/JavaScript .js, VBScript .vbs, PowerShell .ps1, HTA .hta, WSH .wsf и исторически Perl .pl) в пакетный, добавляя в начале каждого файла специальный заголовок-"полиглот", фрагмент кода, синтаксически корректный как в BAT, так и во внедряемом скрипте. В основном, это часть кода, являющаяся комментарием для основного скрипта, использующая некоторые ухищрения для BAT-файла.

Особенности

Для VBS-скрипта дополнительно проверяется наличие директивы Option Explicit, которая "прячется" в комментариях, так как во вновь созданном файле она перестает быть первой исполняемой командой и бросает ошибку компиляции. Хотя наличие этой директивы желательно и даже обязательно (как правильно сказал коллега alexii о правильном стиле программирования), но это требование можно (в данном случае) смягчить в угоду удобству. Предполагается, что исходный vbs-код тщательно отлажен с включенной директивой. Директива должна быть записана в одну строку. Разбиение на несколько строк не поддерживается.

В wsf-файлах используется весьма странная конструкция %~f0?.wsf, которая позволяет запускать bat-файл как wsf-сценарий. Возможно это какая-то недокументированная особенность парсера командной строки. А может быть это бэкдор.

wsf-файл должен начинаться с директивы <?xml?>. Директива должна быть записана в одну строку. Разбиение на несколько строк не поддерживается.

Скрипт не понимает BOM.

Есть возможность выбрать движок для JavaScript, VBScript и WSF с помощью опции /E. Если не укаано явно, то CSCRIPT - движок по умолчанию для JavaScript, VBScript и WSF скриптов. Другое доступное значение - WSCRIPT. Дополнительно для JavaScript файлов можно выбрать другой движок, например, NodeJS, Rhino и т.д. Предопределенная опция /E DEFAULT сбрасывает ранее установленный движок в значение по умолчанию.

Пример работы


:: создать filename1.bat и filename2.bat из исходных файлов
cmdize.bat filename1.vbs filename2.js

Исходный код

+ исходный код cmdize.bat

Скопировать и сохранить в cmdize.bat.
Актуальная версия здесь.


:: USAGE
::     cmdize name [...]
::
:: This tool converts a supported code into a batch file that can be 
:: executed without explicit invoking the executable engine. The script 
:: creates new file and places it under the same directory as the original 
:: one with the same name, replacing the original extension with ".bat". 
:: The content of the new file consists of the original file and the 
:: special header that being the "polyglot" and having some tricks to be a 
:: valid code in batch file and the wrapped code at the same time. 
::
:: FEATURES
:: It does comment on "Option Explicit" in VBScript.
:: "<?xml?>" declaration for wsf-files is expected.
:: "Option Explicit" and "<?xml?>" in a single line only are supported.
:: BOM is not supported at all.
::
:: It is possible to select an engine for JavaScript, VBScript and WSF via 
:: the command line options /E. If it is not pointed especially, CSCRIPT 
:: is used as the default engine for all JavaScript, VBScript and WSF 
:: files. Another valid engine is WSCRIPT. Additionally for JavaScript 
:: files it is possible to set another engine such as NodeJS, Rhino etc. 
:: The predefined option /E DEFAULT resets any previously set engines to 
:: the default value. 
::
:: SEE ALSO
:: Proceed the following links to learn more the origins
::
:: .js
:: http://forum.script-coding.com/viewtopic.php?pid=79210#p79210
:: http://www.dostips.com/forum/viewtopic.php?p=33879#p33879
:: https://gist.github.com/ildar-shaimordanov/88d7a5544c0eeacaa3bc
::
:: .vbs
:: http://www.dostips.com/forum/viewtopic.php?p=33882#p33882
:: http://www.dostips.com/forum/viewtopic.php?p=32485#p32485
::
:: .pl
:: For details and better support see "pl2bat.bat" from Perl distribution
::
:: .sh, .bash
:: http://forum.script-coding.com/viewtopic.php?id=11535
:: http://www.dostips.com/forum/viewtopic.php?f=3&t=7110#p46654
::
:: .ps1
:: http://blogs.msdn.com/b/jaybaz_ms/archive/2007/04/26/powershell-polyglot.aspx
:: http://stackoverflow.com/a/2611487/3627676
::
:: .py
:: http://stackoverflow.com/a/29881143/3627676
:: http://stackoverflow.com/a/17468811/3627676
::
:: .hta and .html?
:: http://forum.script-coding.com/viewtopic.php?pid=79322#p79322
::
:: .wsf
:: http://www.dostips.com/forum/viewtopic.php?p=33963#p33963
::
:: COPYRIGHTS
:: Copyright (c) 2014, 2015, 2016 Ildar Shaimordanov

@echo off

if "%~1" == "" (
	for /f "usebackq tokens=* delims=:" %%s in ( "%~f0" ) do (
		if /i "%%s" == "@echo off" goto :EOF
		echo:%%s
	)
	goto :EOF
)

:cmdize.loop.begin
if "%~1" == "" goto :cmdize.loop.end

if /i "%~1" == "/e" (
	set "CMDIZE_ENGINE="
	if /i not "%~2" == "default" set "CMDIZE_ENGINE=%~2"
	shift
	goto :cmdize.loop.continue
)

if not exist "%~f1" (
	echo:%~n0: File not found: "%~1">&2
	goto :cmdize.loop.continue
)

for %%x in ( .js .vbs .pl .sh .bash .ps1 .py .hta .htm .html .wsf ) do (
	if /i "%~x1" == "%%~x" (
		call :cmdize%%~x "%~1" >"%~dpn1.bat"
		goto :cmdize.loop.continue
	)
)

echo:%~n0: Unsupported extension: "%~1">&2

:cmdize.loop.continue

shift

goto :cmdize.loop.begin
:cmdize.loop.end

goto :EOF


:: Convert the javascript file. 
:: The environment variable %CMDIZE_ENGINE% allows to declare another 
:: engine (cscript, wscript, node etc). 
:: The default value is cscript.
:cmdize.js
if not defined CMDIZE_ENGINE set "CMDIZE_ENGINE=cscript"
set "CMDIZE_ENGINE_OPTS="
for %%e in ( "%CMDIZE_ENGINE%" ) do (
	if /i "%%~ne" == "cscript" set "CMDIZE_ENGINE_OPTS=//nologo //e:javascript"
	if /i "%%~ne" == "wscript" set "CMDIZE_ENGINE_OPTS=//nologo //e:javascript"
)

echo:0^</*! ::
echo:@echo off
echo:%CMDIZE_ENGINE% %CMDIZE_ENGINE_OPTS% "%%~f0" %%*
echo:goto :EOF */0;
type "%~f1"
goto :EOF


:: Convert the vbscript file. 
:: The environment variable %CMDIZE_ENGINE% allows to declare another 
:: engine (cscript or wscript). 
:: The default value is cscript. 
:: The next label is duplicated intentionally to avoid error if this 
:: script was saved in unix mode (EOL=LF).
:cmdize.vbs.h
:cmdize.vbs.h
set /p "=::'" <nul
type "%TEMP%\%~n0.$$"
echo:%*
goto :EOF


:cmdize.vbs
if not defined CMDIZE_ENGINE set "CMDIZE_ENGINE=cscript"

copy /y nul + nul /a "%TEMP%\%~n0.$$" /a 1>nul

call :cmdize.vbs.h @echo off
call :cmdize.vbs.h %CMDIZE_ENGINE% //nologo //e:vbscript "%%%%~f0" %%%%*
call :cmdize.vbs.h goto :EOF

del /q "%TEMP%\%~n0.$$"
rem type "%~f1"
for /f "tokens=1,* delims=]" %%r in ( ' call "%windir%\System32\find.exe" /n /v "" ^<"%~f1" ' ) do (
	rem Filtering and commenting "Option Explicit". 
	rem This ugly code tries as much as possible to recognize and 
	rem comment this directive. It fails if "Option" and "Explicit" 
	rem are located on two neighbor lines, consecutively, one by one. 
	rem But it is too hard to imagine that there is someone who 
	rem practices such a strange coding style. 
	for /f "usebackq tokens=1,2" %%a in ( '%%s' ) do if /i "%%~a" == "Option" for /f "usebackq tokens=1,* delims=:'" %%i in ( 'x%%b' ) do if /i "%%~i" == "xExplicit" (
		echo:%~n0: Commenting Option Explicit in "%~1">&2
		echo:rem To prevent compilation error due to embedding into a batch file, 
		echo:rem the following line was commented automatically.
		set /p "=rem " <nul
	)
	echo:%%s
)
goto :EOF


:: Convert the perl file.
:cmdize.pl
if /i "%CMDIZE_ENGINE%" == "cmd" (
	echo:@echo off
	echo:perl -x -S "%%~dpn0.pl" %%*
	goto :EOF
)

echo:@rem = '--*-Perl-*--
echo:@echo off
echo:perl -x -S "%%~f0" %%*
echo:goto :EOF
echo:@rem ';
echo:#!perl
type "%~f1"
goto :EOF


:: Convert Bourne shell and Bash scripts. 
:cmdize.sh
:cmdize.bash
echo:: ^<^< '____CMD____'
echo:@echo off
echo:bash "%%~f0" %%*
echo:goto :eof
echo:____CMD____
type "%~f1"
goto :EOF


:: Convert the powershell file. 
:cmdize.ps1
echo:^<# :
echo:@echo off
echo:setlocal
echo:set "POWERSHELL_BAT_ARGS=%%*"
echo:if defined POWERSHELL_BAT_ARGS set "POWERSHELL_BAT_ARGS=%%POWERSHELL_BAT_ARGS:"=\"%%"
echo:endlocal ^& powershell -NoLogo -NoProfile -Command "$_ = $input; Invoke-Expression $( '$input = $_; $_ = \"\"; $args = @( &{ $args } %%POWERSHELL_BAT_ARGS%% );' + [String]::Join( [char]10, $( Get-Content \"%%~f0\" ) ) )"
echo:rem endlocal ^& powershell -NoLogo -NoProfile -Command "$input | &{ [ScriptBlock]::Create( ( Get-Content \"%%~f0\" ) -join [char]10 ).Invoke( @( &{ $args } %%POWERSHELL_BAT_ARGS%% ) ) }"
echo:goto :EOF
echo:#^>
type "%~f1"
goto :EOF


:: Convert the python file.
:cmdize.py
:: Ascetic way is shorter but less flexible
:: Uncomment the following 3 lines if it is more preferrable
:: echo:@python -x "%%~f0" %%* ^& @goto :EOF
:: type "%~f1"
:: goto :EOF
echo:0^<0# : ^^
echo:"""
echo:@echo off
echo:python "%%~f0" %%*
echo:goto :EOF
echo:"""
type "%~f1"
goto :EOF


:: Convert the html file. 
:: Supportable file extensions are .hta, .htm and .html. 
:cmdize.hta
:cmdize.htm
:cmdize.html
echo:^<!-- :
echo:@echo off
echo:start "" mshta "%%~f0" %%*
echo:goto :EOF
echo:--^>
type "%~f1"
goto :EOF


:: Convert the wsf file. 
:: The environment variable %CMDIZE_ENGINE% allows to declare another 
:: engine (cscript or wscript). 
:: The default value is cscript.
:cmdize.wsf
if not defined CMDIZE_ENGINE set "CMDIZE_ENGINE=cscript"

for /f "usebackq tokens=1,2,* delims=?" %%a in ( "%~f1" ) do for /f "tokens=1,*" %%d in ( "%%b" ) do (
	rem We use this code to transform the "<?xml?>" declaration 
	rem located at the very beginning of the file to the "polyglot" 
	rem form to do it acceptable by the batch file.
	echo:%%a?%%d :
	echo:: %%e ?^>^<!--
	echo:@echo off
	echo:%CMDIZE_ENGINE% //nologo "%%~f0?.wsf" %%*
	echo:goto :EOF
	echo:: --%%c
	more +1 <"%~f1"
	goto :EOF
)
goto :EOF


rem EOF

Комментарии приветствуются в теме обсуждения.

( 2 * b ) || ! ( 2 * b )

2 (изменено: Rumata, 2015-03-23 03:13:57)

Re: CMD/BAT + JS/VBS/PS1/WSF/HTA/PL/PY/BASH: Два в одном

1. код стал чуточку короче (примерно на 50-60 символов).
2. избавились от неуклюжего трюка с переменной $input: в scriptblock можно передавать напрямую данные из входного потока (с invoke-expression этого нельзя).
3. работая со scriptblock можно описать аргументы и их типы (с invoke-expression нельзя) .
4. можно работать с аргументами командной строки в стиле PowerShell - заключать их как в двойные (стандартный стиль dos) так и в одинарные кавычки (стиль powershell).
5. изменил ссылку на актуальную версию скрипта

( 2 * b ) || ! ( 2 * b )

3 (изменено: Rumata, 2015-05-16 20:51:43)

Re: CMD/BAT + JS/VBS/PS1/WSF/HTA/PL/PY/BASH: Два в одном

Версия со script-block менее работоспособна. Поэтому вернулся к версии с invoke-expression. Аргументация в этом сообщении обсуждений.

Сейчас используется и поддерживается версия с invoke-expression. Версия со script-block оставлена и вставляется в результирующий код в виде комментария. Если поменять комментарий - будет работать script-block версия.

( 2 * b ) || ! ( 2 * b )

4 (изменено: Rumata, 2015-05-30 06:49:09)

Re: CMD/BAT + JS/VBS/PS1/WSF/HTA/PL/PY/BASH: Два в одном

Из обсуждения.
Добавил возможность выбора движка для js/vbs/wsf файлов из CSCRIPT и WSCRIPT.
Для js более актуально указать другой движлк, например, NodeJS. Пример:


// example.js

if ( this.WScript ) {
   WScript.Echo('hello, jscript!');
} else {
   console.log('hellow, nodejs!');
}

C:\>cmdize example.js
C:\>example.bat
hello, jscript!

C:\>cmdize /e node example.js
C:\>example.bat
hello, nodejs!

( 2 * b ) || ! ( 2 * b )

5

Re: CMD/BAT + JS/VBS/PS1/WSF/HTA/PL/PY/BASH: Два в одном

Добавил поддержку Python. Обновленный скрипт в первом сообщении. название темы соответственно поправил.

( 2 * b ) || ! ( 2 * b )

6

Re: CMD/BAT + JS/VBS/PS1/WSF/HTA/PL/PY/BASH: Два в одном

Решился все-таки добавить поддержку shell/bash скриптов - вариант 2 из темы CMD/BAT + BASH: Два в одном. Обновленный скрипт -- в первом сообщении, отразил изменение в названии темы.

( 2 * b ) || ! ( 2 * b )