Rich Lab. Blog

「まつらリッチ研究所」の研究の一環で、はてなブログ上にブログを公開。でもなぜか買い物カゴ(要サードパーティCookie有効化)が……。はてブにはそんな機能あったっけ?

POSIX原理主義の本は、作り方から売り方まで思想の塊である。

この記事はPOSIX原理主義Advent Calendar 2016の2日目の記事です。

今日のテーマは、昨日の記事でも紹介されていたPOSIX原理主義の書籍の話です。いきなりですが、商品の購入ボタンを置いておきます。

f:id:richmikan:20161201224717j:plain

POSIX原理主義聖典

Windows/Mac/UNIX すべてで20年動くプログラムはどう書くべきか

2016年11月10日 発売

頒布価格:3600円

388ページ、2色刷

目次商品詳細

現在、カゴに冊入っています

(Safariをお使いの方はコチラで注文)

続きを読む

Shell Scriptライトクックブック2014-2016 頒布中

2016年の夏コミで発行したShell Scriptライトクックブック2014-2016を通販してます

f:id:richmikan:20160912080501j:plain

POSIX原理主義シェルスクリプトレシピ集

Shell Scriptライトクックブック 2014-2016(第4版)

2016年8月14日 発行(コミケ90 3日目)

頒布価格:1500円

ページが増えて232Pに

目次商品ページ

現在、カゴに冊入っています

(Safariをお使いの方はコチラで注文)

第4版の変更点

詳細は目次をご覧ください。と済ませるのもあんまりなので、ポイントを記します。

Bash on Ubuntu on WindowsTwitterに対応し、青いカバーに

そもそも改訂は第3版でおしまいと言っていたのですが改訂しました(スミマセン)。というのも、Bash on Ubuntu on Windowsという大革命が起きたからです。POSIX原理主義Windowsでも通用する時代が到来したとなったらそれをサポートすべく対応しないわけにはいかない、というわけでBash on Ubuntu on Windowsに対応した内容にしました。

併せてレシピも追加しましたが、一番の目玉はTwitterに関するレシピシェルスクリプトからTwitter APIを操作して、投稿、検索、DM送受信などの日常使いを実現する方法を追加しました。

つまりこれらは、Bash on Ubuntu on Windowsでも動くTwitterアプリが作れるということを意味します。

計15個のレシピが、追加 or 修正

先程紹介したTwitterレシピをはじめ、合計15のレシピを追加しました。5章の「どの環境でも動くシェルスクリプトを書く」の内容をより完璧にしたり、他に大きなものとしては、1秒未満sleepをいかにしてPOSIXの範囲で実現するかなどを解説しています。たぶんこれはコロンブスの卵的に目から鱗なはずです。

序章を完全に書き直し、POSIX原理主義の理論をより完璧なものに

POSIXの範囲では不可能な処理を行うためにはどうしてもPOSIX外のコマンドを利用しなければなりませんが、従来は「やむを得ないので、最低限にとどめる」という見解であり、理論的に不完全でした。第4版ではこれを修正、「交換可能性の担保」という考え方を導入することで、理論的な曖昧さを解消しました。

また、東京メトロのコンテストから1年半経過し、一体どれだけの応募作品が今でも動いているのかの実態調査レポートも掲載しました。POSIX原理主義を用いないソフトウェアの保存の利かなさが露呈する結果になりました。(こんな結果で東京五輪観光客向けのアプリ開発促進計画は大丈夫なんでしょうか?)

通販もやってます

通常のコミケではないし……、遠いし……、行けないよ。という方もご安心を。通販やってます。この通販のためのプログラムは、本書で紹介したレシピをふんだんに活用し、シェルスクリプトで作られたものです。本書レシピの実力の程を確認するためにも、このサイトの通販を是非ご利用ください。(Safari等、このサイトのカゴ入れボタンがうまく機能しない方はコチラへ)

 

 

値引きをプログラミングする(売り方も作品の一部だ)

POSIX原理主義者割引き」はじめました

f:id:richmikan:20150719144455j:plain

書籍「すべてのUNIXで20年動くプログラムはどう書くべきか」を著者の通販サイト(ココのサイト)で購入されるお客様に、ささやかながら値引きを致します。

とはいっても商業本を値引きすることはできない決まりなので、元になった同人誌をセットで購入される方に、その同人誌の方を400円値引きします。

他の書店には絶対マネできないこの企画。是非下のボタンを2つともポチしてください。(Safariの方は本店でポチ!

商業版カゴ入れボタン 同人版カゴ入れボタン

冊入っています

冊入っています

「分割金利手数料はすべて負担」はジャパネットたかたの『作品』

以前、あのジャパネットの高田社長が「ショッピングはエンターテインメントです」と言っていた。「分割金利手数料は店が負担」の第一人者だし、ハプニング覚悟で生放送を止めないし、高田社長は経営を引退しても未だに出演は続けている。

それらはまさに、エンターテインメントだという前提があるからであって、ジャパネットたかたはメーカーが作った作品(=製品)を売りながら、自分達が考案したショウ(=実演)や値引き(=分割金利手数料負担然り)という作品を売っているのだ。

つまり「値引き」とは店の作品の一つである。作品であるということは、値引きの仕方にもオリジナリティーがなくてはならない。客はオリジナリティーを感じるから、例え値引き額がさほどでなくても、その店を選ぶ。(実際、ジャパネットの価格も価格.comで見つかる最安店には敵わないことが多い。それでも売れる)

ネットショップ店主が独自の値引き企画を創設するのは難しいという現実

そうすると「なるほど、独自の値引き制度を考案するのも一つの手なのか!よし、やろう」と思う。ところが、多くのネットショップ店主にとってはそれが難しい。なぜなら、ネットショップ上で値引きを実施したければ、ショッピングカートプログラムにやらせなければならない。でも店主は基本的にプログラマーではないので、ショッピングカートで値引き販売をしたければ既製のものから選ぶしかない。結局それは店独自の「作品」ではない。

WordPressといったCMSなど、ページデザイン(店の内装)の制作にはある程度店独自にオリジナリティーを表現させるプログラムはあるのに、なぜ値引き制度という分野ではオリジナリティーを表現させるプログラムがないのか!

値引きルールをプログラミングで表現する

このような問題点に気付いたので私は自前のショッピングカートプログラムに値引き機能を実装してみた。もちろん、既製の値引き機能を作るのが目的ではない。導入する店主が独自の値引き企画を導入できる仕組みだ。

細かい話はヌキにして、どうやったかだけを記そうと思う。ちなみに、うちの自作ショッピングカートはシェルスクリプトで書かれている。

1) 買い物明細生成プログラム本体にフックを置く

値引き機能を入れ込む対象となるのは、買い物明細を生成するプログラムである。しかし、これを下手いじるとシステムそのものが復旧不可能なほどに壊れてしまうという問題がある。

そこで、開発者と店主がいじるべき領域の違いを明確化すべく、店主がいじるべきプログラムファイルは外出しし("$Homedir/CONFIG"というディレクトリーの中に"DISCOUNT_RULES.SHLIB"という名前で置くこととし)、存在すればインクルード(".":sourceコマンド)するようにした。→冒頭部分

######################################################################
# 初期設定
######################################################################

# --- このシステム(CART)のホームディレクトリー -----------------------
Homedir="$(d=${0%/*}/; [ "_$d" = "_$0/" ] && d='./'; cd "$d.."; pwd)"

# --- その他初期ディレクトリー等定義 ---------------------------------
[ -f $Homedir/CONFIG/DISCOUNT_RULES.SHLIB ] && { # 値引きルールライブラリー登録
  . $Homedir/CONFIG/DISCOUNT_RULES.SHLIB         # (存在すれば)
}
   :
   :
   :

# --- 小計を求め、レシートファイルに記入 -----------------------------
cat $Tmp-in_the_cart                                                            |
# 1:商品ID 2:カゴ入り数量                                                       #
join -1 1 -2 1 -o 2.1,1.2,1.3,1.4,1.5,1.6,1.7,2.2 $Tmp-productmaster -          |
# 1:商品ID 2:商品名(mojihameエンコ) 3:単価(+税表記含) 4:発売日 5:重さ厚さ       #
# 6:在庫情報体 7:URL 8:カゴ入り数量                                             #
if PATH='' type DISCOUNT_BEFORE_CALCULATING_TAX >/dev/null 2>&1; then           # 税金計算前の値引き
  DISCOUNT_BEFORE_CALCULATING_TAX                                               # (ルールが設定されている場合)
else                                                                            #
  cat                                                                           #
fi                                                                              |
awk '{print $1,$2,$7,$3,$8}'                                                    |
# 1:商品ID 2:商品名(mojihameエンコード) 3:URL 4:単価(+税表記含) 5:カゴ入り数量  #
awk 'BEGIN          {sotozei_target_amt = 0;                          }         #
     $4~/\+[Tt]$/   {zeinuki_unitprice = substr($4,1,length($4)-2);             #
                     sotozei_target_amt += zeinuki_unitprice*$5;                #
                     print $1,$2,$3,zeinuki_unitprice,$5,zeinuki_unitprice*$5;} #
     $4~/^-?[0-9]+$/{print $0,$4*$5;                                  }         #
     END            {tax = int(sotozei_target_amt*'"$TR_TAX_RATE"');            #
                     print "TAX 消費税(税別商品分) -", tax, 1, tax;}'           > $Tmp-reciept
# 1:商品ID 2:商品名(mojihameエンコード) 3:URL 4:単価(税抜) 5:カゴ入り数量 6:小計

上記のコードの後半部分は、明細テキストデータを生成している箇所だ。joinやawkをパイプで繋ぐといういかにもシェルスクリプトらしいコードである。

このコードの中盤に"DISCOUNT_BEFORE_CALCULATING_TAX"という謎のコマンドが出てくる。これが(消費税計算前に)値引きを行うためのプログラム(後述)なのだが、じつは外部コマンドではなくてシェル関数である。typeコマンドによってその存在を確かめてから、存在すれば実行し(パイプで通し)、存在しなければcatコマンドで到来データをそのまま通過させるというロジックだ。

if文ごとパイプに通せるのがいかにもシェルスクリプトでしょ?

2) 店主がいじる値引き実行用のシェル関数

明細生成プログラム本体はフックを入れただけなので、値引きを実行するプログラム本体は別にある。店主は、独自に考案した値引き企画を、この部分でシェルスクリプトという表現手段を用いて表現する。

店主は、値引きを企画したいのであれば $Homedir/CONFIG/DISCOUNT_RULES.SHLIB という名のファイルを用意し、DISCOUNT_BEFORE_CALCULATING_TAXというシェル関数を書く。

今回は、商業本(商品ID=pfb)と同人本(商品ID=ssr2)を同時購入した筋金入りのPOSIX原理主義者に向けた割り引きだが、それをプログラミングしたものが次のシェルスクリプトだ。

######################################################################
# 商品確認後、税計算前段階での値引き
######################################################################
#
# [使用方法]
# * 使いたい場合は、このシェル関数の名前を
#   "DISCOUNT_BEFORE_CALCULATING_TAX"
#   にして、下記の入出力の規則に則ってプログラムを書く。
# [入力]
# * 標準入力から次のデータが送られてくる
#   1:商品ID
#   2:商品名(mojihameエスケープ)
#   3:単価(外税商品は末尾に"+t"と書かれている)
#   4:発売日
#   5:重さ厚さ情報
#   6:在庫情報体(電子商品は"*")
#   7:商品URL(なければ"-")
#   8:カゴ入り数量
# [出力]
# * 標準出力に上記情報をそのまま通過させるのを忘れずに!!!!!
# * それに加えて、必要に応じて次の列を出力する。
#     1:値引ID("^DISC.[A-Za-z0-9_.-]*$"形式に合致すること。例、"DISC.campain1")
#     2:値引理由の名称(mojihameエスケープ)
#     3:値引単価
#       - 負値にすること!そうしなければ割増しになってしまう
#       - 外税商品の値引きをする場合は数字の直後に"+t"を付けると税も値引かれる
#     4:"-"(ダミー列につき無視されるので何でもよい)
#     5:"-"(ダミー列につき無視されるので何でもよい)
#     6:"-"(ダミー列につき無視されるので何でもよい)
#     7:値引理由を説明しているページがあればそのURL(無ければ"-")
#     8:適用数(第4列の名目を複数口適用したい場合はここを2以上にする)
#
DISCOUNT_BEFORE_CALCULATING_TAX() {

  awk '
  {
    # カゴに入っている商品IDとその単価・数量の登録をしながら出力
    pri[$1] = $3;
    qty[$1] = $8;
    print;
  }
  END {

    # 「POSIX原理主義者」割り引き
    # ・商品ID[ssr2]が存在する場合、商品ID[pfb]1冊につき400円値引く。
    # ・ただし値引き額の最大は、商品ID[ssr2]の小計である。
    if (("ssr2" in qty) && ("pfb" in qty)) {
      p = -400      ;
      q = qty["pfb"];
      if (p*q*(-1) > pri["ssr2"]*qty["ssr2"]) {
        p = pri["ssr2"]*qty["ssr2"]*(-1);
        q = 1;
      }
      print "DISC.POSIX POSIX原理主義本同時購入よる同人版値引き",p,"- - - -",q;
    }

  }
  '
}

実体はAWKのコードである。それがシェル関数でラッピングされているに過ぎない。

標準入力から、冒頭のコメントにある通りの仕様でカゴに入っている商品明細データが渡ってくるのでそれを標準出力へと素通りさせつつ、そのうしろに今回の値引きルールに基づく値引き項目を追加して標準出力へ送るのである。出力時のデータフォーマットも、冒頭に記されたコメントに従う。

値引きのためのプログラミング学習も、デザインのためのHTML学習と同程度にしたい

上記の解説はプログラミングに慣れた人にとってはさほど難しい話ではない。では、そうではない純粋な商売人にはどうだろうか?

確かに上記のようにして値引き実施コードを本体から切り離したものの、きちんと書かなければ本体も動かなくなるわけで完全に安全になったとは到底言えない。

しかし、ネットショップの内装(デザイン)を弄るためにHTMLやCMSを覚えことのがある程度受け入れられているのだから、その程度にまで敷居を下げさえすればよいと思うし、またできるはずだ。

それができなければ、個人のネットショップなど大手に潰されてしまう。

 

 

 

 

とある日本の自動車メーカーの社用車がベンツだった件と、シェルスクリプトの実用性を謳う本の通販サイトがPHPだった件の関係性

日本の自動車メーカーの社用車がベンツ?

「総務部総務課山口六平太」というマンガ知ってますか? 自動車会社に勤めるサラリーマンが主人公の話ですけど、日本の自動車メーカーなのに社用車がベンツというのがネタだったのです。

「車売ってる会社なのに、社長はベンツに乗ってんのかよ!」って、誰しもツッコミたくなりますよね。まぁ、あれは大日自動車株式会社という架空の会社ですけど、じゃあ実在の日本の自動車会社はどうかというと、あれ?いすゞ!。……まぁ、今や乗用車は作っていないですからね。(そもそも情報古いですし)

メーカー 氏名 社長車 私用車
トヨタ 張 富士夫 センチュリークラウン プログレコロナ
ホンダ 吉野 浩行 レジェンド ラグレイト
日産 カルロス・ゴーン プレジデント セドリックエルグランド
マツダ マーク・フィールズ センティア RX-7
三菱 園部 考 ディグニィディ ギャランVR-4
ダイハツ 山田 隆哉 アルティス アルティス
スバル 竹中 恭二 レガシィB4 レガシィB4
スズキ 戸田 昌男 カルタスセダン なし
いすゞ 井田 義則 キャデラックコンコース ビックホーンアスカ

車の常識(15@僕のくるまさんのページより

自動車会社にとって社長の車といったら社の象徴なので、それが他社製だったら身も蓋もない話ですよね。

コンピューター言語の実用性を謳う本の販売サイトが他言語製

ところが、コンピューター言語においては、そんな身も蓋もない話が、誰にもツッコミを受けることなく行われてきました。シェルスクリプトアプリ開発言語としての実用性を謳う本の通販サイトがPHPとか……、そりゃないでしょ。

いゃベンツだってPHPだっていい車や言語ですし、それに別に売るなってわけじゃないですよ。でも著者が、他言語で作られた販売サイトを宣伝する(しかできない)って何のジョークだ!と思うのです。

山口六平太の会社の社用車も自社製になった

でも、彼の会社の社用車がベンツだったのは初期の話。念願かなって今ではサザンという自社製のセダンになっています。

一方シェルスクリプト本は……?

つい昨日、念願かなって、シェルスクリプト製になりました。いや、今までも既に販売サイト自体はあったんですけど、今回めでたく商業本を私(著者)手作りのシェルスクリプト製販売サイトで取り扱えることになりました。C&R研究所さん、ありがとう!こういう柔軟さは大手には簡単には真似できませんね。

というわけでカゴ入れボタンを貼っておきますね。

f:id:richmikan:20150707222712j:plain

すべてのUNIX20年動くプログラムはどう書くべきか

~デプロイ・保守に苦しむエンジニア達へ送るシェルスクリプトレシピ集~

リアル リッチ・ミカン[著]、USP研究所[監]、C&R研究所[発行]

2015年7月24日 発売

本体2,500+税200円

A5版 2色刷 0x100頁

目次商品ページ

現在、カゴに冊入っています

(Safariをお使いの方はコチラ)

このサイト(はてなブログ)自体は、まず間違いなくシェルスクリプト製ではないでしょうけど、ショッピングカートのパーツ(カゴ入れ&レジボタン)はシェルスクリプト製ですからね。このショッピングカートプログラムの特長は、例え他言語CMSで作られたページでもショッピングサイトと化すことができるという点なのです。

だから、人気の出たブログ上でブログ主が商売も始めたいと思ってもわざわざ引っ越す必要がなくなるのです。便利でしょ?

ちなみに、本家の商品ページは現在、ウザい宣伝欄がくっついています(笑)。これもこのショッピングカートプログラムが本質的にCMSを選ばないから簡単にできる技。

 

 

 

 

POSIX原理主義のユリカモメ本、商業本になる

前回の冬コミとコミスペ6で頒布したPOSIX原理主義同人誌が、商業本になって本屋に並ぶことになりました。

f:id:richmikan:20150707222712j:plain

すべてのUNIX20年動くプログラムはどう書くべきか

~デプロイ・保守に苦しむエンジニア達へ送るシェルスクリプトレシピ集~

リアル リッチ・ミカン[著]、USP研究所[監]、C&R研究所[発行]

2015年7月24日 発売

本体2,500+税200円

A5版 2色刷 0x100頁

目次商品ページ

現在、カゴに冊入っています

(Safariをお使いの方はコチラ?)

超硬派なUNIX哲学&POSIX原理主義

どれくらい硬派かというとbashbashという言語であって、シェルスクリプトを語るに値せず」というくらい硬派です。そう言うと反感を覚える人もいるでしょうが、bash独自拡張機能を利用しておきながら冒頭に平気で「#! /bin/sh」を書く人に対して反感を覚えるという人なら、きっと仲良くなれるはずです。

Bourne Shellを至上とし、もちろんGNUコマンド等も使わず(使ってもいいけど独自拡張機能は一切使わず)、極力POSIXの範囲でプログラムを組みます。それなのに、こーーんなことがちゃんとできます。

さようなら。デプロイ沼とメンテナンス地獄

実用性があることはわかったとして、なぜわざわざPOSIX原理主義にこだわるのか?それは「どのUNIXに持っていってもそのまま動く」、「10年20年の長きに渡ってそのまま動く」という、超可搬性・超可用性を追求するためです。

世の中、「速い」「簡単」を謳う言語やライブラリー、フレームワークで溢れてます。コストを削減せよ!業務を効率化せよ!とウンザリするほど言われているとそういうものに飛びつきたくなります。しかし飛びついたらどうなるか……。待ち構えているのはメンテナンス地獄です。

  • サポート終了といわれて、面倒なバージョンアップを強いられた。
  • 脆弱性が見つかって、緊急バージョンアップを迫られた。
  • 言われてバージョンアップしたらシステムが動かなくなった!

とかそんなのばっかり。本当にコスト削減&業務効率化は図られているのかと、疑いたくなります。

一方でPOSIX原理主義を遵守すると、こういう目にはほとんど遭わずに済むようになります。なにせPOSIXは枯れているからです。枯れているゆえに殆ど機能追加はありませんし、ましてや機能の廃止など殆どありません。とは言え「OSに脆弱性が見つかったらメンテしないわけには行かないだろ」と思うかもしれません。いや大丈夫です、どのUNIX系OS開発団体もPOSIX規格だけは遵守しようとしますから、POSIXの範囲で組んだプログラムしかないから、何も恐れることなくOSのバージョンアップができます。また、OSを乗り換えることもできます。しかもシェルスクリプト+テキストファイルでシステム組んでいれば、コピーするだけで乗り換え完了です。

さようなら、デプロイ沼とメンテナンス地獄……。

ところでこの商業本、Qiita投稿と同人誌が元なんです

f:id:richmikan:20150324165114j:plain

冒頭でも出てきた同人誌というのは、これです。あ、同人誌ですからね、コレ。某出版社のページ探しても見つかりませんよ。ちなみに、このイラストはウチのサークルで手描きしたものです。

内容は同じで、旧バージョン(最新版は2015/03/29版)です。ただ、今回商業本化するにあたっては、クォリティーの違いを見せつけられましたね。実物を比べてみれば明らかに商業本の方が欲しくなるデキです。

同人誌もQiitaへの投稿が元

商業本と同人誌本はクォリティーの違いで棲み分けができていると思うのですが。実は、同人誌の元ネタ(つまり商業本の元ネタでもある)は、私がQiitaに書き貯めた投稿です。

中には商業本化の話が動き出してから投稿した記事もあります。先にWebに公開して、それを後から本に収録したのです。

「そんなことをしたら本が売れなくなるでしょ?」と疑問に思うかもしれませんが、私の経験上そんなことはありません。事実、この同人誌はWebに公開せずに作った同人誌(シリーズ第一弾)と同じ勢いで売れましたし、中には「Web(の記事)でいつもお世話になってまーす」と言って買いにいらしたお客さんもいました。

ですのでこの本(同人誌&商業本)には、Webで同じ記事があっても売り上げには影響しないことを実証する意味合いがあります。むしろ、Webで事前公開することによって記事のオープンβテストができるという利点さえあります。

どうでしょう、Qiitaでほぼ同じ内容が見られると知ったら購買意欲を削がれましたか?

さてさて、この本はどれだけ売れるでしょうか?内容に興味の無い方も、その結果にご注目を!

 

 

 

Shell Script ライトクックブック2014 第二版

昨年末の冬コミで発行したShell Scriptライトクックブック2014、第二版をコミケットスペシャルで発売します。

f:id:richmikan:20150324165114j:plain

POSIX原理主義シェルスクリプトレシピ集

Shell Scriptライトクックブック 2014 第二版

2015年3月28日 発行(コミケットSP6 1日目)

頒布価格:1200円

ページが増えて154Pに

目次商品ページ

現在、カゴに冊入っています

(Safariをお使いの方はコチラで注文)

第二版の変更点

詳細は目次をご覧ください。と済ませるのもあんまりなので、ポイントを記します。

計10個のレシピが、追加 or 修正

シェルスクリプトを書いていると次から次へとレシピが増えますが、今回も少し増えました。例えば、Base64エンコードとデコードです。Linux等ではBase64のコーデックたるbase64コマンドが始めから入っていて不要なレシピかもしれませんが、POSIXの範囲で実装した話を紹介しています。これによって、他の環境でも安心してBase64が扱えます。

Base64との相互変換を実現するは、バイナリーコードを扱える必要があります。ところが、AWKの変数(GNU版除く)やシェル変数(zsh除く)ではそれができません。では一体どうやってシェルスクリプトバイナリーコードを扱うのか、という話もレシピとして追加しました。

その他、どの環境でも使えるシェルスクリプトを書くためのレシピを多数追加加筆しました。

より強化したPOSIX原理主義

ところで、なぜBase64を扱うレシピを追加したのか。それは、メール送信のレシピからnkfコマンドを葬り去るためです。

これまでは、UTF-8文字をSMTPサーバーが許容する7ビット文字セットであるJISコードに変換するためにnkfを利用していました。しかし、Base64エンコードをして英数字だけにしてしまえばUTF-8のまま送ることが可能になるというわけです。また、Subject:などに置く日本語文字列をBエンコードISO-2022-JPBase64エンコード)するためにもnkfコマンドを利用していましたが、これもUTF-8のままBase64変換すればよいわけで、base64コマンドという車輪の再発明によってこちらからも排除することに成功しました。

さようならnkf……、他にもmktempやwhichをPOSIXの範囲で実装することに成功したため、それらに頼っていたレシピもよりPOSIX原理主義忠実なものになりました。

コミケットスペシャル6で初売り

完成した第二版は、3/28(土)から開催されるコミケットスペシャル6で頒布します。スペース番号は、1日目 K-58bです。

通販もやってます

通常のコミケではないし……、遠いし……、行けないよ。という方もご安心を。通販やってます。この通販のためのプログラムは、本書で紹介したレシピをふんだんに活用し、シェルスクリプトで作られたものです。本書レシピの実力の程を確認するためにも、このサイトの通販を是非ご利用ください。(Safari等、このサイトのカゴ入れボタンがうまく機能しない方はコチラへ)

 

 

東京メトロの列車接近情報を返すWebAPIをシェルスクリプトで叩く

既にいろいろなところで話に上っているけど、東京メトロ「オープンデータ活用コンテスト」というのをやっている。

時刻表や駅の設備といった静的な情報はもちろん、リアルタイムの列車位置なんかも公開するという大盤振る舞い。「これで役立つアプリを作ってね」ということなのだが、じゃあ俺がシェルスクリプトで何か作ってやろうじゃないかと、いうことでパイプを駆使した活用プログラムを作ってみた。

接近情報表示コマンド「メトロパイパー」

f:id:richmikan:20140921114151j:plain

一般的な意味の「パイパー」とは、
こういう人のことを言うらしい

というわけで作ってみたプログラム、その名は「メトロパイパー」。パイパーというと、本来は右の写真のような男の娘を意味するらしいのだが、ここでのパイパーとは「UNIXシェルのパイプを操る者」と「地下鉄のパイプ(=路線)を覗く者」という意味にしている。

各種情報はここ以外にも下記のサイトに置いておいたので参考にしてね。

ところで「接近情報」って?

まず接近情報とは何かだが、駅のホームへ行くとそこの電光掲示板に表示される、

【こんど】快速 東葉勝田台
【つぎ 】各停 西船橋
※ 「こんど」&「つぎ」なんていつの時代だ

とか

前々駅 -◆- 前駅 --- 当駅

というアレだ。

でも、あの情報がその駅に居なくても見られたら便利だよなー、とまえまえから思っていた。例えばこんなシーンはないだろうか。

  • 時間ギリギリにホームに着いたけど、目的の電車が来ない!遅れてるのかな?それともやっぱり出発しちゃったのかな?(あぁ、先の駅の接近情報が見られれば……)
  • 急行南栗橋行間に合わなかったなぁ……。次の東武線直通はどの駅まで来てるんだろう。
  • あ、特急ロマンスカー霞ヶ関駅あたりに来てるな。よし、じゃあそろそろ帰宅の準備して表参道駅へ行くか。

という具合に、今いる駅の、先や、もっと手前の駅の接近情報、あるいは駅にいなくてもこれから行く予定の駅の接近情報ってわかれば便利でしょ。使い方は直感的にわかるように作っているつもりなので使ってみてね。

UNIX哲学とPOSIX原理主義溢れるソースコード

このプログラムの特徴は何と言っても、中身のほとんど全て*1シェルスクリプト+UNIX標準コマンドで書かれているということだ。しかもUNIX哲学の思想を随所に採り入れ、パイプを多用している点だ。

そのいくつかを紹介しよう。

JSONテキストを行列指向に変形して捌く!

まずはソースコード「SHELL/VIEW_METROLOC_0.SH」を見てもらいたい。

######################################################################
# 列車ロケーション情報API呼び出しと、キャッシュ作成
# (キャッシュの有効期限が切れているならば)
######################################################################
:
# --- 有効キャッシュがなければ作る(タイムスタンプは有効期限日時に設定)
[ $cache_is_fresh -eq 0 ] && { curl -s $url | parsrj.sh > "$File_cache"; }
:
:

######################################################################
# API取得データ(列車ロケーション情報の取得)の加工
######################################################################

# --- 必要な項目・区間&方面のみに絞り込んだ在線位置マスターファイルを作る
cat "$File_cache" |
sed 's/^\$\[\([0-9]\{1,\}\)\]\.[^:]\{1,\}:/\1 /' |
awk '$2=="railDirection" {$2=1;print;} #
$2=="fromStation" {$2=2;print;} #
$2=="toStation" {$2=3;print;} #
$2=="trainType" {$2=4;print;} #
$2=="terminalStation"{$2=5;print;} #
$2=="trainOwner" {$2=6;print;}' |
sort -k1n,2n |
awk '{print $3}' |
awk 'NR%6!=0{printf("%s ",$0)} NR%6==0{print $0}' |
# 1:方面コード 2:発車駅コード 3:到着駅コード 4:種別コード 5:目的駅コード 6:車両所有業者コード
:

これは実際にAPIを叩いて値を解析しているコードの一部だ。

東京メトロの今回のオープンデータAPIJSON-LDという形式で情報を返してくる。そこで拙作のJSONパーサー"parsrj.sh"(→解説)で、(1)値の位置、(2)値という、各行2列で構成されるテキストに変換し、それをさらにAWKを使って(1)方面コード(上り下り的なもの)、(2)発車駅コード、(3)到着駅コード、(4)種別コード、(5)目的駅コード、(6)車両所有業者コード、という6列のデータに変換している。

こうやってJSONを一旦行列指向のフォーマットに変換してしまえば、あとはgrepAWKで絞り込んだり加工したり……と、リレーショナルデータベース的な操作が自由にできる。

JOINコマンドでマスターデータをハメ込む

「リレーショナルデータベース的な操作」と言ったからには当然JOINもできる。例えば、同じソースコード「SHELL/VIEW_METROLOC_0.SH」の下記の部分を見てもらいたい。

# --- コードを名称化し、列車存在位置から列車名を引くマスターファイルを作る
cat $Tmp-this-rw-dir-loc |
# 1:現在居る3桁駅ナンバー(3桁目は0または5で、5は中間にいることを表す) 2:種別コード
# 3:目的駅コード 4:車両所有業者コード #
sort -k2,2 |
join -1 1 -2 2 -a 2 -o 2.1,2.2,1.2,2.3,2.4 $Homedir/DATA/METRO_VOC_MST.TXT - |
sed 's/\([^. ]\{1,\}\) /\1 \1 /' |
awk '{print $1,$3,$4,$5}' |
# 1::現在居る3桁駅ナンバー(3桁目は0または5で、5は中間にいることを表す) 2:種別名
# 3:目的駅コード 4:車両所有業者コード #
sort -k3,3 |
join -1 1 -2 3 -a 2 -o 2.1,2.2,2.3,1.2,2.4 $Homedir/DATA/METRO_VOC_MST.TXT - |
sed 's/\([^. ]\{1,\}\) /\1 \1 /' |
awk '{print $1,$2,$4,$5}' |
# 1:現在居る3桁駅ナンバー(3桁目は0または5で、5は中間にいることを表す) 2:種別名
# 3:目的駅名 4:車両所有業者コード #
sort -k4,4 |
join -1 1 -2 4 -a 2 -o 2.1,2.2,2.3,2.4,1.2 $Homedir/DATA/METRO_VOC_MST.TXT - |
sed 's/\([^. ]\{1,\}\) /\1 \1 /' |
awk '{print $1,$2,$3,$5}' |
# 1:現在居る3桁駅ナンバー(3桁目は0または5で、5は中間にいることを表す) 2:種別名
# 3:目的駅名 4:車両所有業者名 |
sort -k1,1 > $Tmp-this-rw-loc

APIから送られてくる接近情報データは人間に分かりやすい名称ではなく、各種コード文字列になっている。これを人間向けにするなら、各種コードに対応する名称の記されたマスターファイルとJOINして、名称の列に置換する必要がある。

それをやっているのがこのコードだ。ここでは、種別(各停とか急行とか)、目的駅、車両所有業者を、各々コードから名称にするため、マスターファイル(METRO_VOC_MST.TXT)と3回JOINしているのがわかるだろう。

Ajaxで送るWebパーツは、JSONではなくてHTMLコードそのもの

上記はデータの加工に関する話であったが、Webインターフェースを扱う部分にも特徴が表れている。

動作デモを見てきてもらえばわかるが、駅を選択する<select>タグの中身はAjaxで書いている。大抵は、Webサーバーから<option>タグに渡す名称と値をJSONなりXMLなりで受け取り、項目の数だけdocument.createElementをforループで回すことだろう。

しかし、サーバーサイドがシェルスクリプトならではの技で、<option>タグHTMLそのものをサーバー側で生成し、<select>エレメントのinnerHTMLにゴッソリ流し込むというやり方を使っている。

まず、流し込まれるHTMLテンプレート「TEMPLATE.HTML/MAIN.HTML」を見てもらいたい。中に次のような記述の箇所がある。

  :
<td>
<select id="from_snum" name="from_snum" disabled="disabled" onchange="get_locinfo()">
<!-- FROM_SELECT_BOX -->
<option value="-">選んでください</option>
<!-- FROM_SNUM_LIST
<option value="%1">%1 : %2線-%3駅</option>
FROM_SNUM_LIST -->
<!-- FROM_SELECT_BOX -->
</select>
</td>
:

このHTMLファイルは当然サーバー上に置かれており、次のシェルスクリプト「CGI/GETSNUM_HTMLPART.AJAX.CGI」でこれを利用している。その一部を抜粋する。

# --- 部分HTMLのテンプレート抽出 -------------------------------------
cat "$Homedir/TEMPLATE.HTML/MAIN.HTML" |
sed -n '/FROM_SELECT_BOX/,/FROM_SELECT_BOX/p' > $Tmp-htmltmpl

# --- HTML本体を出力 -------------------------------------------------
cat "$Homedir/DATA/SNUM2RWSN_MST.TXT" |
# 1:駅ナンバー(sorted) 2:路線コード 3:路線名 4:路線駅コード
# 5:駅名 6:方面コード(方面駅でない場合は"-")
awk '{print substr($1,1,1),$0}' |
sort -k1f,1 -k2,2 |
awk '{print $2,$4,$6}' |
uniq |
# 1:駅ナンバー(sorted) 2:路線名 3:駅名 #
grep -i "^$rwletter" | # ←関係ある路線だけに絞り込んでいる
mojihame -lFROM_SNUM_LIST $Tmp-htmltmpl -

まず前述のHTMLのうち<option>タグの部分(FROM_SELECT_BOXというコメントで囲まれた区間)をsedで抽出している。その後、駅名マスターファイル(SNUM2RWSN_MST.TXT)の中に書いてある、(1)駅ナンバー、(2)路線名、(3)駅名のみをAWKなどで取り出し、mojihameというコマンドに流している。

ここで呼び出しているmojihameというは、やはりシェルスクリプトで書かれている自作コマンドなのだが、標準出力から与えられた行数だけ、先程抽出したテンプレートの<option>タグの行を複製するという処理を行っている。<option>タグの箇所には"%1"~"%3"というマクロ文字があるが、これらが標準入力から渡ってきた各列の文字列に置換される。これによって、

 <!-- FROM_SELECT_BOX -->
<option value="-">選んでください</option>
<option value="C01">C01 : 千代田線-代々木上原駅</option>
<option value="C02">C02 : 千代田線-代々木公園駅</option>
<option value="C03">C03 : 千代田線-明治神宮前〈原宿〉駅</option>
:
<option value="Z14">Z14 : 半蔵門線-押上〈スカイツリー前〉駅</option>
<!-- FROM_SELECT_BOX -->

というテキストが生成されるのだ。あとは、先程言ったようにこれをAjaxブラウザーに渡して、innerHTMLでごっそりハメ込んでもらえば完成。

シェルスリプトなんて非実用的でしょ?

実はこのプログラムは、そんな先入観を吹っ飛ばすべくして作った。

シェルスクリプトのポータビリティーは実は高い!

なにせこのプログラムは、curlコマンドとWebサーバー(Apache)以外は全てPOSIXの範囲で作っている。LAMPという言葉があるが、つまりMとPの部分は無し!だからLとAのある環境ならどこでも動くということだ。そしてコンパイルも不要、POSIXの範囲だからBash依存やGNU依存なども無い。さらに言うと、LとAの部分の実装は問わないので、Lの代わりはFreeBSDでも、Solarisでも構わないし、Aの代わりにnginxなどでも構わない。シェルスクリプトは高いポータビリティーを持つ、ということだ。

シェルスクリプトだって短期間開発ができる

また、「できるのはわかるけど、フレームワーク無しに一から作ってたら時間かかるだろうが」と思うかもしれないが、それも誤りだ。このコンテスト募集開始日(2014/09/12)から、本ソフトの公開日(2014/09/16)までたった4日であることに注目すべきだ。しかも実際は1日で書いている(信じないかもしれないが)。公開日が16日なのは、ソースコードの公開が利用規約に違反しないという公式回答が15日にあったので、「それなら作るか!」と作ったのが始まりだったのだ。1日、それが信じられなくても4日で作ったというこの事実が、シェルスクリプトは短期開発でも「使える」ということを証明しているだろう。

 

見直した? 見直したら、そんなシェルスクリプトでとあるアプリを開発したとうネタを収録した薄い本


コミケ4日目は、はてブで本を買おう - りっちけんきゅうじょのにっき

も、よろしくね。(ちゃっかり宣伝)

 

 

*1:curlコマンド以外

←カゴに商品が入っている時にこのボタンを押せば、レジへ移動します。(カゴに商品を入れるにはサードパーティCookieを有効化する必要あり)