<?xml version="1.0" encoding="UTF-8"?>
<feed xmlns="http://www.w3.org/2005/Atom" xmlns:dw="https://www.dreamwidth.org">
  <id>tag:dreamwidth.org,2011-12-22:1181104</id>
  <title>Блокнотик Centaur’а</title>
  <subtitle>Yuri Khan</subtitle>
  <author>
    <name>Yuri Khan</name>
  </author>
  <link rel="alternate" type="text/html" href="https://yurikhan.dreamwidth.org/"/>
  <link rel="self" type="text/xml" href="https://yurikhan.dreamwidth.org/data/atom"/>
  <updated>2017-09-28T13:13:18Z</updated>
  <dw:journal username="yurikhan" type="personal"/>
  <entry>
    <id>tag:dreamwidth.org,2011-12-22:1181104:70831</id>
    <link rel="alternate" type="text/html" href="https://yurikhan.dreamwidth.org/70831.html"/>
    <link rel="self" type="text/xml" href="https://yurikhan.dreamwidth.org/data/atom/?itemid=70831"/>
    <title>Подсветка синтаксиса высших порядков</title>
    <published>2017-09-28T13:13:18Z</published>
    <updated>2017-09-28T13:13:18Z</updated>
    <category term="json"/>
    <category term="usability"/>
    <category term="syntax"/>
    <category term="yaml"/>
    <category term="xml"/>
    <dw:security>public</dw:security>
    <dw:reply-count>0</dw:reply-count>
    <content type="html">&lt;p&gt;В старые добрые времена, когда людям нужно было выразить что-нибудь в компьютерном файле, они изобретали новый синтаксис. Потом в чью-то умную голову пришла мысль подсвечивать элементы синтаксиса в текстовых редакторах.&lt;/p&gt;

&lt;p&gt;Потом люди придумали универсальные языки разметки — XML, JSON и YAML — с помощью которых можно выразить всё что угодно. Поэтому теперь вместо синтаксиса часто изобретают схему.&lt;/p&gt;

&lt;a name="cutid1"&gt;&lt;/a&gt;
&lt;p&gt;Однако, что у нас с их подсветкой? Возьмём, к примеру, кусочек мета-схемы JSON Schema Draft 04 (который примечателен тем, что описывает свою собственную структуру):&lt;/p&gt;

&lt;pre style="background-color: white; color: black"&gt;&lt;code&gt;{ &lt;span style="color: #4e9a06"&gt;"$schema"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"http://json-schema.org/draft-04/schema#"&lt;/span&gt;,
  &lt;span style="color: #4e9a06"&gt;"type"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"object"&lt;/span&gt;,
  &lt;span style="color: #4e9a06"&gt;"properties"&lt;/span&gt;: {
    &lt;span style="color: #4e9a06"&gt;"$schema"&lt;/span&gt;: {&lt;span style="color: #4e9a06"&gt;"type"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"string"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"format"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"url"&lt;/span&gt;},
    &lt;span style="color: #4e9a06"&gt;"$ref"&lt;/span&gt;: {&lt;span style="color: #4e9a06"&gt;"type"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"string"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"format"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"url"&lt;/span&gt;},
    &lt;span style="color: #4e9a06"&gt;"type"&lt;/span&gt;: {&lt;span style="color: #4e9a06"&gt;"type"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"tsring"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"enum"&lt;/span&gt;: [
      &lt;span style="color: #4e9a06"&gt;"null"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"boolean"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"integer"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"number"&lt;/span&gt;,
      &lt;span style="color: #4e9a06"&gt;"string"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"array"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"object"&lt;/span&gt;]},
    &lt;span style="color: #4e9a06"&gt;"format"&lt;/span&gt;: {&lt;span style="color: #4e9a06"&gt;"type"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"string"&lt;/span&gt;},
    &lt;span style="color: #4e9a06"&gt;"properties"&lt;/span&gt;: {&lt;span style="color: #4e9a06"&gt;"type"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"object"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"aditionalProperties"&lt;/span&gt;: {&lt;span style="color: #4e9a06"&gt;"$ref"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"#"&lt;/span&gt;}},
    &lt;span style="color: #4e9a06"&gt;"additionalProperties"&lt;/span&gt;: {&lt;span style="color: #4e9a06"&gt;"$ref"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"#"&lt;/span&gt;},
    &lt;span style="color: #4e9a06"&gt;"enum"&lt;/span&gt;: {&lt;span style="color: #4e9a06"&gt;"type"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"array"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"items"&lt;/span&gt;: {&lt;span style="color: #4e9a06"&gt;"$ref"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"#"&lt;/span&gt;}},
    &lt;span style="color: #4e9a06"&gt;"items"&lt;/span&gt;: {&lt;span style="color: #4e9a06"&gt;"$ref"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"#"&lt;/span&gt;}
  }
}&lt;/code&gt;&lt;/pre&gt;

&lt;p&gt;Одни сплошные «строковые литералы». В лучшем случае — ключи объектов одним цветом, остальные строки другим. На первом плане — синтаксис самого JSON; а более высокоуровневый синтаксис, построенный поверх JSON’а, остаётся нераскрашенным и непроверяемым на глаз. Найдите две опечатки в примере выше.&lt;/p&gt;

&lt;p&gt;Надо — имея на руках схему, валидировать об неё документ и подсвечивать по-разному:&lt;/p&gt;

&lt;ul style="background-color: white"&gt;
&lt;li style="color: #5c3566"&gt;ключи, явно допустимые в текущем контексте;&lt;/li&gt;
&lt;li style="color: #204a87"&gt;ключи, допустимые по шаблону или по принципу «всё, что не запрещено»;&lt;/li&gt;
&lt;li style="color: #a40000"&gt;ключи, недопустимые в текущем контексте, и примитивные значения, не соответствующие схеме;&lt;/li&gt;
&lt;li style="color: #8f5902"&gt;значения, явно соответствующие заданным ограничениям;&lt;/li&gt;
&lt;li style="color: #4e9a06"&gt;все остальные строки.&lt;/li&gt;
&lt;/ul&gt;

&lt;pre style="background-color: white; color: black"&gt;&lt;code&gt;{ &lt;span style="color: #5c3566"&gt;"$schema"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"http://json-schema.org/draft-04/schema#"&lt;/span&gt;,
  &lt;span style="color: #5c3566"&gt;"type"&lt;/span&gt;: &lt;span style="color: #8f5902"&gt;"object"&lt;/span&gt;,
  &lt;span style="color: #5c3566"&gt;"properties"&lt;/span&gt;: {
    &lt;span style="color: #204a87"&gt;"$schema"&lt;/span&gt;: {&lt;span style="color: #5c3566"&gt;"type"&lt;/span&gt;: &lt;span style="color: #8f5902"&gt;"string"&lt;/span&gt;, &lt;span style="color: #5c3566"&gt;"format"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"url"&lt;/span&gt;},
    &lt;span style="color: #204a87"&gt;"$ref"&lt;/span&gt;: {&lt;span style="color: #5c3566"&gt;"type"&lt;/span&gt;: &lt;span style="color: #8f5902"&gt;"string"&lt;/span&gt;, &lt;span style="color: #5c3566"&gt;"format"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"url"&lt;/span&gt;},
    &lt;span style="color: #204a87"&gt;"type"&lt;/span&gt;: {&lt;span style="color: #5c3566"&gt;"type"&lt;/span&gt;: &lt;span style="color: #a40000"&gt;"tsring"&lt;/span&gt;, &lt;span style="color: #5c3566"&gt;"enum"&lt;/span&gt;: [
      &lt;span style="color: #4e9a06"&gt;"null"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"boolean"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"integer"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"number"&lt;/span&gt;,
      &lt;span style="color: #4e9a06"&gt;"string"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"array"&lt;/span&gt;, &lt;span style="color: #4e9a06"&gt;"object"&lt;/span&gt;]},
    &lt;span style="color: #204a87"&gt;"format"&lt;/span&gt;: {&lt;span style="color: #5c3566"&gt;"type"&lt;/span&gt;: &lt;span style="color: #8f5902"&gt;"string"&lt;/span&gt;},
    &lt;span style="color: #204a87"&gt;"properties"&lt;/span&gt;: {&lt;span style="color: #5c3566"&gt;"type"&lt;/span&gt;: &lt;span style="color: #8f5902"&gt;"object"&lt;/span&gt;, &lt;span style="color: #204a87"&gt;"aditionalProperties"&lt;/span&gt;: {&lt;span style="color: #5c3566"&gt;"$ref"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"#"&lt;/span&gt;}},
    &lt;span style="color: #204a87"&gt;"additionalProperties"&lt;/span&gt;: {&lt;span style="color: #5c3566"&gt;"$ref"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"#"&lt;/span&gt;},
    &lt;span style="color: #204a87"&gt;"enum"&lt;/span&gt;: {&lt;span style="color: #5c3566"&gt;"type"&lt;/span&gt;: &lt;span style="color: #8f5902"&gt;"array"&lt;/span&gt;, &lt;span style="color: #5c3566"&gt;"items"&lt;/span&gt;: {&lt;span style="color: #5c3566"&gt;"$ref"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"#"&lt;/span&gt;}},
    &lt;span style="color: #204a87"&gt;"items"&lt;/span&gt;: {&lt;span style="color: #5c3566"&gt;"$ref"&lt;/span&gt;: &lt;span style="color: #4e9a06"&gt;"#"&lt;/span&gt;}
  }
}&lt;/code&gt;&lt;/pre&gt;&lt;br /&gt;&lt;br /&gt;&lt;img src="https://www.dreamwidth.org/tools/commentcount?user=yurikhan&amp;ditemid=70831" width="30" height="12" alt="comment count unavailable" style="vertical-align: middle;"/&gt; comments</content>
  </entry>
  <entry>
    <id>tag:dreamwidth.org,2011-12-22:1181104:66551</id>
    <link rel="alternate" type="text/html" href="https://yurikhan.dreamwidth.org/66551.html"/>
    <link rel="self" type="text/xml" href="https://yurikhan.dreamwidth.org/data/atom/?itemid=66551"/>
    <title>CLI как язык</title>
    <published>2016-08-22T11:01:24Z</published>
    <updated>2016-08-22T11:01:24Z</updated>
    <category term="cli"/>
    <category term="syntax"/>
    <category term="usability"/>
    <category term="soft"/>
    <category term="lxc"/>
    <dw:security>public</dw:security>
    <dw:reply-count>4</dw:reply-count>
    <content type="html">&lt;p&gt;Синтаксис аргументов командной строки следует рассматривать и проектировать аналогично синтаксису человеческого языка. В частности, в нём бывают различные части речи и члены предложения.&lt;/p&gt;

&lt;p&gt;Название бинарника может быть глаголом в повелительном наклонении (&lt;code&gt;reboot&lt;/code&gt;) или существительным в роли обращения (&lt;code&gt;firefox&lt;/code&gt;, &lt;code&gt;git&lt;/code&gt;). В последнем случае отсутствие аргументов соответствует запуску приложения с пользовательским интерфейсом; если аргументы есть, то первый — это опять глагол в повелительном наклонении (&lt;code&gt;git fetch&lt;/code&gt;).&lt;/p&gt;

&lt;p&gt;Ключ без аргумента соответствует наречию, обычно в роли обстоятельства образа действия (&lt;code&gt;--quietly&lt;/code&gt;, &lt;code&gt;--verbosely&lt;/code&gt;; по истерическим перчинам устоялось написание без суффикса -ly).&lt;/p&gt;

&lt;p&gt;Ключ с аргументом — это косвенное дополнение, где имя ключа играет роль предлога (&lt;code&gt;install &lt;b&gt;-t /usr/bin&lt;/b&gt; foo&lt;/code&gt; — &lt;i&gt;установить &lt;b&gt;в /usr/bin&lt;/b&gt; foo&lt;/i&gt;); или уточняющее родовое слово в составе прямого дополнения (&lt;code&gt;install &lt;b&gt;-d /var/lib/foo&lt;/b&gt;&lt;/code&gt; — &lt;i&gt;установить &lt;b&gt;каталог /var/lib/foo&lt;/b&gt;&lt;/i&gt;).&lt;/p&gt;

&lt;p&gt;Позиционный аргумент — это прямое дополнение (&lt;code&gt;git clone git://github.com/git/git.git&lt;/code&gt;).&lt;/p&gt;

&lt;p&gt;Это, конечно, не все паттерны — что-то я наверняка упустил.&lt;/p&gt;

&lt;p&gt;Базовый язык для команднострочного интерфейса — разумеется, английский. (Если бы командную строку изобрёл японец, глагол ставился бы последним, а &lt;code&gt;sudo&lt;/code&gt; записывалось бы как 下さい [kudasai] после глагола.)&lt;/p&gt;

&lt;p&gt;Собственно, я это всё к чему? У системы виртуализации/контейнеризации LXC есть команды &lt;code&gt;lxc-start&lt;/code&gt;, &lt;code&gt;lxc-stop&lt;/code&gt; и несколько других. И все они принимают название контейнера, над которым работать, именованным аргументом (&lt;code&gt;lxc-start -n foo&lt;/code&gt;). Жутко бесит. Очевидно же, что это должно быть прямое дополнение.&lt;/p&gt;&lt;br /&gt;&lt;br /&gt;&lt;img src="https://www.dreamwidth.org/tools/commentcount?user=yurikhan&amp;ditemid=66551" width="30" height="12" alt="comment count unavailable" style="vertical-align: middle;"/&gt; comments</content>
  </entry>
</feed>
