Вы не находите странным черту освистывать нечто до появления некого более функционального аналога оного и проявлять к этому нечто интерес после? На протяжении всего своего существования командная оболочка Windows неоднократно подвергалась жесткой критике, дескать, ее функциональность оставляет желать лучшего, что, казалось бы, должно было сойти на нет с появлением PowerShell, призванного устранить недочеты первой и упростить жизнь разработчикам и системным администраторам. Нет, PowerShell снискал должную популярность, но появился интерес и к самой командной строке особенно после того, как «селекционерами» были открыты способы запускать командные сценарии как WS[H|F] и HTA. Собственные эксперименты и наблюдения показали, что этим дело не ограничивается.

Некоторые символы в cmd наделены особой логикой, например, двоеточие в начале команды означает метку, а потому если командный сценарий состоит из единственной строки :label или даже одного двоеточия, интерпретатор просто ее проигнорирует; если после двоеточия стоит один из знаков перенаправления потока, сути это также не изменит — интерпретатор по-прежнему будет считать данную строку меткой, что в общем-то логично. Вообще, изменяя порядок использования некоторых спецсимволов командной строки, можно добиться любопытных побочных эффектов, таких как, скажем, запуск PHP кода в виде командного сценария:

:<?php /*
@echo off
  echo:Start
  2>nul php "%~f0" %*
  echo:End
exit /b
*/
echo "\r";

foreach (array_slice($argv, 1, count($argv)) as $i) echo "$i\n";
?>

Или то же, но — bash:

:<<EOF
@echo off
  echo:Start
  2>nul bash "%~f0" %*
  echo:End
exit /b
EOF
for i in $@;do echo $i;done

Конечно же при этом обыгрываются особенности самих вызываемых интерпретаторов, но основная суть в том, что такое вообще возможно. Возможен также и запуск сценариев Perl, Ruby или Python как командных сценариев в виду наличия у перечисленных интерпретаторов ключа -x, заставляющий последние игнорировать первую строку передаваемого на исполнение кода. Строго говоря, данное утверждение справедливо для Python, а вот Perl и Ruby с данным ключом будут игнорировать все до тех пор, пока не встретят шебанги #!perl и #!ruby соответственно.

Perl:
@echo off
  echo:Start
  2>nul perl -x "%~f0" %*
  echo:End
exit /b
#!perl
foreach my $i (@ARGV) {print $i, "\n";}

Ruby:
@echo off
  echo:Start
  2>nul ruby -x "%~f0" %*
  echo:End
exit /b
#!ruby
ARGV.each do |i| puts i;end

Python:
@echo off&echo:Start&2>nul python3.4m -x "%~f0" %*&echo:End&exit /b
from sys import argv
for i in range(1, len(argv)): print(i)

Подобный подход применим и к PowerShell, а вот утверждать, что описанное выше применимо и к другим интерпретируемым языкам, не стану, могу лишь допустить такую возможность причем с совсем другого ракурса.

Комментарии (8)


  1. Regis
    01.05.2016 15:14
    +4

    Экое извращение. Не дай бог такое «in the wild» встретить!


  1. MooNDeaR
    01.05.2016 15:26
    +24

    Как связан заголовок статьи, первый абзац и какая-то кодо-дичь?


    1. Lertmind
      01.05.2016 20:23
      +1

      Автор сделал вывод: после выхода PowerShell, cmd немножко стал популярнее (то есть его не забыли), что даже нарыли не задокументированную хитрость. Конечно, ерунда это всё: bat файлы как использовали, так и используют.


      1. denis_g
        01.05.2016 23:48

        Некоторые особо упорные даже игры на этом пишут.


        1. IvanAnonym
          04.05.2016 01:09
          +1

          И «антивирусы»…


  1. Shtucer
    01.05.2016 17:40
    +4

    Статья про то, что cmd нужен лишь для того, чтобы запустить любым способом нормальный интерпретатор? Удивительное открытие.


  1. DuMOHsmol
    01.05.2016 18:19

    А вот аналогичный трюк для C#
    http://pastebin.com/embed_iframe/T56TiUNd
    Когда-то давно это было на хабре, но потом та статья куда-то исчезла.


    1. Lertmind
      01.05.2016 20:37
      +2

      Пользователь удалён/деактивирован. В archive.org/web/ не сохранилась, но например здесь есть копипаста с хабра с объяснением.