Promise and Generator #7

さて、wasavi の根幹の動作ロジックを Promise ベースに大改造した大元の動機のひとつである、migemo 対応について考えてみたい。

とりあえず Chrome extension としての migemo についてまとめてみたい。migemo 自体のホームはここ。そして Chrome 版の migemo というのは、検索文字列に対応する migemo 的な正規表現を返す API を提供することに特化した拡張で、migemo server ということになっている。

まず edvakf さんが制作した ChromeMigemo Extension というものがかつてあった。これ自体はすでにサポートを終了しているが、ソースは公開されている。ただ困ったことに API リファレンスといったドキュメントはないので、このリポジトリ自体はエンドユーザにとってはそれほど有用ではない。API リファレンス的なものは辛うじて彼のはてなダイアリーの古い記事から得られる。

さて、これを現在サポートしているのは mono さんで、blog の記事でfork 版である Migemo Server for Google Chrome™の告知がなされている。wasavi もこれを参照する。

これを wasavi の [cci]/[/cci]、[cci]?[/cci] コマンドに組み込むわけだが。もともとこれらのコマンドは正規表現を入力して検索するものだ。これが migemo に対応すると、入力した文字列が内部で複雑な正規表現に変換された後に検索が行われるという流れになる。つまり何か入力してそれを検索するという手順は同じなのだが、入力するものの性格は全く違うということだ。migemo の場合は入力するのは文字列リテラルなのだ。

そうなると、コマンド自体を別に起こす(例えば、[cci]g/[/cci]、[cci]g?[/cci] とか)ほうが筋がいいのかもしれない。しかし、ここはやはり [cci]/[/cci]、[cci]?[/cci] コマンドにまとめたい。

ところで vim では、正規表現中に [cci]\C[/cci] というものを含めると ignorecase オプションの値にかかわらず、必ず大文字・小文字を区別して検索するようになる。[cci]\C[/cci] には正規表現のメタキャラクタとしての効果は何もなく、それが正規表現中に存在するか否かに意味があるというわけだ。この考え方を流用してみよう。メタキャラクタ [cci]\M[/cci] を導入し、これが検索文字列に含まれている場合は migemo で検索、そうでない場合は従来の検索というように振り分ける。

このやりかたにすると、検索文字列の入力中に「やっぱ migemo で検索しよーっと」とかその逆が簡単に切り替えられるし、また [cci]map g/ /\\M[/cci] などとすることも可能なので、migemo 検索を独立したコマンドにしたいという向きにも対応できる。

というわけで、できた。

Marquee!

wasavi が起動する際、最小の横幅・縦幅というものがあり、横の最小幅は 320px ということになっている。

このくらい幅が狭いと、ステータスラインに何かメッセージを表示する際に、その長さによっては後ろが隠れてしまうことがある。

これを解決するために、いわゆる marquee 的な動作をさせるようにしてみた。的な、というか、実際に marquee 要素で囲むのである。21 世紀も17年過ぎて marquee 要素を使うとは思わなかった…。

この marquee 要素、いつ廃止されてもおかしくない状況なのだが、とりあえず今のところは Chrome でも Firefox でも使える。ただし Chrome においてはかなりスムーズに動くのに対して Firefox ではかなりガックガクだ。

まああんまり気にしないことにしよう。

Promise and Generator #6

一通り修正が終わって、Selenium による Chrome 上の機能テストも一応クリアした。

一応というのは、800 超の機能テストはカテゴリごとに editing.js とか、insert.js とか分かれているのだが。それらを個別に動かすと 100% パスするものが、全てまとめて通すとぽろぽろと失敗するものがランダムに出てくる。これはなんだろうか。

もうひとつ。以前 Firefox でテストしようとしたところ、geckodriver の機能や、Selenium のサポートなどがまだ熟成されていない感じだった。やたらめったら CPU パワーを消費するし、既存のプロファイルを使用できないし、WebExtensions ベースの拡張を認識してくれないし、要素に対して sendKeys() する機能が未実装だし…など、プロダクトレベルの品質に達しているとはとても言えない状態だった。あれから数ヶ月経ったがどうだろうか。

ということで試してみたところ、
Could not convert 'text' to string
なるエラーが sendKeys() で発生する。うーんまだ実装していないのかな? と調べてみたところ、issue が上がっていた。つまるところ、sendKeys() 自体は実装済みなのだが、Selenium は入力されるべきキー情報を文字の配列の形で送出するのに対し、geckodriver 側は文字列の形で受けることを想定しているという型のミスマッチのようだ。issue 後半に掲載されているクイックパッチの通り直してみたらとりあえず動いた。javascript バインディングの Selenium の場合、スクリプト言語で実装されているおかげでこういう小回りがきくのは嬉しい。

その他、CPU パワーを浪費することもないようで、さすがにいろいろと進歩しているようだ。

Mount a windows share folder #3

CIFS で Windows の共有ディレクトリをマウントすると Linux 側が不安定になる件。結局 nfs サーバは試していない。いや一応 WinNFSd だけ試してみたのだが、ファイル名のエンコーディングを指定できずに化けてしまうのにがっくりしてそこから追ってない。

Linux 側の現象を見てみるに、どうも一旦スリープして、復帰するとマウントの状態がおかしくなっている気がする。一旦おかしくなると Thunar は固まるし、端末からマウントしたディレクトリを覗いただけで固まって困る。

ということはスリープ時に一旦アンマウントして、復帰したらマウントしなおせばいいのか?

と思ってググってみたらそのものズバリのケースがあった。これもまた割と既知の現象だった。/etc/pm/sleep.d/ に以下のようなシェルスクリプトを置いておく:

#!/bin/sh
# Unmount CIFS share on hibernate/suspend and remount it on resume

case "$1" in
hibernate|suspend)
umount /media/windows/music
umount /media/windows/pictures
;;
thaw|resume)
mount /media/windows/music
mount /media/windows/pictures
;;
*) exit $NA
;;
esac

なるほどねえ。

ちなみにおかしくなったマウントの状態は [cci]umount -l /media/windows/music[/cci] などと -l オプションをつければ一応解消できるのだが。このオプションは、つまりアンマウントのための様々なクリーンアップは後回しにして、とりあえずファイルシステムの階層からの切り離しを優先するとのことだ。後回しにしただけなので、もしかしたらマウントがおかしくなった状態を完全に解消する方法とは言えないのかもしれない。

Promise and Generator #5

ex の executor はかなり書き換えたわけだが、一方で vi コマンドの executor というのも一応ある。これは ex ほど大規模ではなくて、単に vi コマンドの文字列をキー入力キューに挿入するだけだ。この機能は [cci].[/cci] コマンド及び [cci]@[/cci] コマンドが利用している。つまりいわゆるマクロ機能に相当する。

今回コマンドの実行が非同期になったことで難しくなったのは、マクロの実行が完了したタイミングが取れないということで


executeViCommand('2w');
doSomethingAfterExecute();

ということができない。そこで、キー入力の最後に特別な擬似キー [cci]*macro-end*[/cci] を付加するようにした。なぜマクロ完了のタイミングが重要なのかといえば、カウントというものがあるからだ。[cci].[/cci] の場合は実行する vi コマンド列がカウントを内包しているのであんまり関係ないのだが、[cci]@[/cci] の場合はそうではない。

そういうわけでこのマクロ実行完了を表す擬似キーを使う。対応するコマンドハンドラ内でカウントを見て、必要なら再度コマンド列をキー入力キューに充填したりすればいい。ところでそのハンドラは normal モードにしか存在しないので、マクロ実行完了時のモードもまた normal モードじゃいけない。つまり他のモードの場合は [cci]escape[/cci] キーを再充填するなどの処置も必要。

それから、[cci]@[/cci] コマンドというのは指定したレジスタの内容を vi コマンドとして実行する機能なのだが。そうすると今回施した修正の元、たとえば [cci]@a[/cci] という文字列がレジスタ [cci]a[/cci] に格納されている状態で [cci]@a[/cci] を実行すると実質的な再帰が起こってしまう。ちなみに vim 7.4 (2013 Aug 10, compiled Nov 24 2016 16:44:48) で試したところ、固まってしまった(固まらない時もある。よくわからない)。

というわけで、[cci]@[/cci] コマンドのもとで vi コマンドを実行している際は同じレジスタを使った [cci]@[/cci] コマンドを入れ子で使えないようにした。入れ子でなければいいので、@a が @b を呼び、それが @c を呼び…ということはできる。

Promise and Generator #4

ぼちぼち Selenium でテストし始める。

ところで Selenium でテストする際、ブラウザの console 出力を得たい時がある。…のだけど、いまいちそのへんのドキュメントが整備されていない気がする。javascript バインディングの API はドキュメントがあるが、これはあくまでリファレンスであってハウツー的なものはほとんど書いてない。

いろいろ調べてみたところ、

const webdriver = require('selenium-webdriver');
const chrome = require('selenium-webdriver/chrome');
:
:
var loggingPrefs = new webdriver.logging.Preferences()
loggingPrefs.setLevel(
webdriver.logging.Type.BROWSER,
webdriver.logging.Level.ALL);

var options = new chrome.Options();
options.setLoggingPrefs(loggingPrefs);

var driver = new webdriver.Builder()
.withCapabilities(webdriver.Capabilities.chrome())
.setChromeOptions(options)
.build()
:
:
driver.manage().logs().get(webdriver.logging.Type.BROWSER).then(log => {
// log は Entry インスタンスの配列で、
// 各要素の message プロパティに console 出力の各行が格納されている
});

という感じでいいようだ。message プロパティは前後にソースファイル名や行番号などが付加されていたりするので、そういうものを削ぎ落とす加工は必要。

Early arousal syndrome

Windows10マシンが、いつのまにかスリープさせると数秒後に勝手に復帰し、さらに数秒後にまたスリープし、そして再び数秒後に勝手に復帰…というのを繰り返すようになった。おもしろいので1時間くらい放っておいたが1時間くらい繰り返している。

イベントビューアからカスタムビューを生成して Power-Troubleshooter を抽出し、復帰の理由を見てみたところ、犯人は VBoxSVC.exe のようだった。つまり VirtualBox だ。お前かよ。お前かよ。

わりと既知の問題のようだ。

とりあえず VirtualBox のバージョンを最新にしたら今のところ収まったように見えるが、また再発するかもしれない。ちなみに復帰の理由はコマンドプロンプトから [cci]powercfg -lastwake[/cci] でも確認できる。

npm 5

npm の CEO である Isaac Schlueter 氏から直々にメールが来て(と言っても単に npm ユーザ全員に送ってるだけだが)、npm が version 5 になったよ! パフォーマンスとか可用性とか色々良くなったから使ってみてね! ということであった。

へー、と [cci]npm install npm@latest -g[/cci] と打ち込んでバージョンが上がったのを確認し、そのまま [cci]npm outdated -g[/cci] としたところ、出て来るべきものが何も出てこない。[cci]npm update -g[/cci] としても何も起こらない。

なんか、バグみたいだ。うーん。テストとかしないのかな。

* * *

件の issue に書かれているワークアラウンドを試してみたところ、さらになんかおかしい。

各モジュール自身のディレクトリがある。例えば [cci]~/.nvm/versions/node/v7.2.1/lib/node_modules/chromedriver[/cci] とか。これに対して [cci]npm install -g chromedriver[/cci] とする。すると、モジュール自身のディレクトリは消え失せ、その代わりに自分自身を指す間違ったシンボリックで置き換えられた。もちろんこれには全く意味はなく、当然モジュールも動作しない。

あらかじめ各モジュールのディレクトリを削除した上で再インストールすればいいようだ。うーんなんか…なんか…。

Promise and Generator #3

長い時間をかけてちくちくいじっていたものが終結しつつある。キーボード入力から vi コマンド / ex コマンドの実行、その実行の完了を保証するシーケンスポイントまでの一連の関数呼び出しが Promise の連鎖によって行われるように修正した。とても疲れた。

これにより、従来あんまり綺麗な形で実装していなかった非同期処理を、それを必要とする関数内に局所的に閉じ込める形にできて見通しが良くなった。例えばクラウドに対する読み書きの場合は、それを担当するのは ex コマンドのハンドラなのだが。応答が複数回来るので、一旦途中で ex 仮想マシンの実行を中断し、wasavi 側のグローバルなメッセージハンドラで受けて、適宜再開する…のようなことをしていた。バックエンドに対するメッセージは基本的に送信と応答の 1 回のやり取りで完結するので複数のやり取りに対応できないからだ。とても不自然。

これを、ex コマンドハンドラ内で Promise を生成し、新規に開いたメッセージポートを通して複数回のやり取りをし、すべてが完了したら resolve() するようになった。とても見通しがいい。

ex コマンドのエグゼキュータも Promise ベースで動作するように書き直した。疲れた。

マップマネージャもまた Promise を使用するように完全に書き直した。これもまた疲れた。

疲れすぎてもう海に還りたい。

Using redshift

いま使っているマシンは Thinkcentre なんとかという、いわゆる一体型 PC なのだが。このマシンを仮に明るい時間にいじっている際は、ディスプレイの輝度はそれなりに上げてほしい。しかし逆に、夜更けにいじっている際は輝度は抑えてほしい。このあたりを賢く調節してくれないだろうか。

というわけで redshift を導入する。~/.config/redshift.conf を以下のような形で適当に書き(経度・緯度は伏せたが、そのとおりの数値を書く)
[redshift]
temp-day=5500K
temp-night=5000K
transition=1
gamma=1.0:0.9:0.9
location-provider=manual
brightness-day=1.0
brightness-night=0.5

[manual]
lat=*
lon=*

これをログイン時に常駐させると、時間帯によってディスプレイの色温度やガンマ値や輝度を適切なものに調整してくれる。すごい。

これを導入する際に、ディスプレイ側の設定が必要になるかもしれない。つまりディスプレイ側で色温度やガンマ値や輝度をハードウェア的に設定できる場合は、それらをデフォルト…というか「なんにも弄らないモード」に戻したり、輝度の場合は最大に設定したりしないといけない。この辺の、ディスプレイ設定のハードウェアとソフトウェアのシームレスかつ密な連携ってどうなっているんだろうか。

* * *

ソースを見てみたら、redshift は色温度・ガンマ値・輝度の3要素を制御できるわけだが、最終的にはすべての要素を掛けあわせたガンマ値にして設定しているようだ。つまりそれぞれ色温度テーブルから求めたガンマ値、R/G/Bに個別に指定できるガンマ値、R/G/B一律に指定するガンマ値という扱いになっている。