Windows環境での開発が苦痛に感じてきたので、仮想マシンのUbuntuで開発できないかを試してみました。

Table of Contents

はじめに

ホストマシンの環境

  • Windows10 Home 10.0.16299
  • Vagrant 1.9.5
  • VirtualBox 5.1.30 r118389

ゲストマシン(Ubuntu)の環境

  • Ubuntu16.04
  • 日本語対応版

Ubuntu Desktopの作成

VagrantとVirtualBoxを使ってUbuntu Desktopを作成します。

Vagrantfile

必須の箇所は必ず記載が必要です。
任意設定は必要であれば記載してください。

# -*- mode: ruby -*-
# vi: set ft=ruby :

Vagrant.configure("2") do |config|
  # ************************************
  # 必須!
  # ************************************
  config.vm.box = "bento/ubuntu-16.04"

  # ************************************
  # 任意設定
  # ************************************
  # port forward
  # ホストマシンからアクセスしたいポート番号を +10000 でフォワードする
  # ex. Windowsの10080 => Ubuntuの80
  ([80, 1313, 3000, 6006, 8000] + (8080..8090).to_a).each do |port|
    config.vm.network "forwarded_port", guest: port, host: port + 10000, host_ip: "127.0.0.1"
  end
  config.vm.network "private_network", ip: "192.168.33.10"

  # ************************************
  # 必須!
  # ************************************
  # VirtualBox VM options
  config.vm.provider "virtualbox" do |vb|
    # 好きな名前とスペックを設定
    vb.name = "ubuntu-gui"
    vb.cpus = "2"
    vb.memory = "4096"
    # GUI settings
    vb.gui = true
    vb.customize [
      "modifyvm", :id,
      # ビデオメモリの割り当て
      "--vram", "256",
      # 3DアクセラレータをONにしないと画面がカクカクになる
      "--accelerate3d", "on",
      "--hwvirtex", "on",
      "--nestedpaging", "on",
      "--largepages", "on",
      "--ioapic", "on",
      "--pae", "on",
      "--paravirtprovider", "kvm",
      # クリップボードやドラッグ&ドロップをWindowsと共有
      "--clipboard", "bidirectional",
      "--draganddrop", "bidirectional",
      # 仮想マシンのモニタ数。2にしたら描画がもっさりしたので泣く泣く1に....
      "--monitorcount", "1",
    ]
  end

  # ************************************
  # 任意設定
  # ************************************
  # sync
  # Windowsとマウントしたい場合は設定する
  config.vm.synced_folder "../../", "/vagrant"
  config.vm.synced_folder "../../../../", "/whome"

  # ************************************
  # 必須!
  # ************************************
  # Provision OS
  # Desktop環境 + 日本語環境 + timezoneとlocaleの設定
  # OSをバージョンアップするまでは二度と実行しない
  config.vm.provision "shell", inline: <<-SHELL
    # Install ubuntu-desktop
    sudo apt -y update
    # See https://github.com/chef/bento/issues/661
    # sudo apt-get -y upgrade だと interactiveな画面でCUIが固まってしまうため以下のコマンドを代わりに実行
    DEBIAN_FRONTEND=noninteractive apt-get -y -o Dpkg::Options::="--force-confdef" -o Dpkg::Options::="--force-confold" upgrade
    sudo apt-get -y dist-upgrade
    sudo apt-get -y install ubuntu-desktop

    # Install for japanese
    wget -q https://www.ubuntulinux.jp/ubuntu-ja-archive-keyring.gpg -O- | sudo apt-key add -
    wget -q https://www.ubuntulinux.jp/ubuntu-jp-ppa-keyring.gpg -O- | sudo apt-key add -
    sudo wget https://www.ubuntulinux.jp/sources.list.d/xenial.list -O /etc/apt/sources.list.d/ubuntu-ja.list
    sudo apt -y update
    sudo apt-get -y upgrade
    sudo apt-get -y install ubuntu-defaults-ja

    # Set timezone and locale
    sudo timedatectl set-timezone Asia/Tokyo
    sudo localectl set-locale LANG=ja_JP.utf8
  SHELL

  # ************************************
  # 任意設定
  # ************************************
  # Install docker
  # DockerをVagrantのprovision機能でインストールする
  config.vm.provision "docker"

  # ************************************
  # 任意設定
  # ************************************
  # Install ansible and provisioning
  # AnsibleをVagrantのprovision機能でインストールする
  # 設定内容はtadashi-aikawaのものであり、各自設定する必要がある
  config.vm.provision "ansible_local" do |ansible|
    ansible.playbook = "ansible/site.yml"
    ansible.inventory_path = "ansible/local"
    ansible.limit = "localhost"
    ansible.raw_arguments = ["-K"]
    ansible.extra_vars = {
      env: 'vm-gui'
    }
  end

  # ************************************
  # 任意設定
  # ************************************
  # Install jenkins
  # 個人的にJenkinsをインストールしたいので追加
  # Ansibleでインストールしても良い。(面倒だったのでVagrant provisioningに記載した)
  config.vm.provision "shell", inline: <<-SHELL
    wget -q -O - https://pkg.jenkins.io/debian/jenkins-ci.org.key | sudo apt-key add -
    sudo sh -c 'echo deb http://pkg.jenkins.io/debian-stable binary/ > /etc/apt/sources.list.d/jenkins.list'
    sudo apt-get -y update
    sudo apt-get -y install jenkins
  SHELL
  
end

上記設定ファイルで使用しているAnsibleのPlaybookは以下になります。
本記事執筆時とは内容が変わっている可能性がありますのでご了承下さい。

実行

上記のVagrantfileを実行します。

$ vagrant up --provision

かなり時間がかかりますので、その間にVagrantfile内の解説をご覧下さい。
処理が終わったらvagrantから再起動します。

$ vagrant halt && vagrant up

Ubuntuから再起動するとボリュームマウントが上手くいきませんでした。

ログインパスワードは vagrant です。

日本語キーボードを使用する

デフォルトでは英語版のキーボードになっています。
それが良い場合は問題ありませんが、日本語版が使いたい場合は以下の設定をします。

AutoHotKeyを自然に使えるようにする

私の場合はAutoHotKeyをバリバリ使っておりますので以下の設定が必要でした。
ホストOSでキーフックを設定していない場合は設定不要だと思います。

キーボードの自動キャプチャーをオフにする

キーボードのキャプチャーとはVirtualBoxの機能です。
VirtualBoxのウィンドウがアクティブなとき、キーボードからの入力を直接ゲストに送る機能だと思います。

以下が点灯しているとき、キーボードのキャプチャーはONになっています。
キャプチャーのON/OFFはホストキー(デフォルト: 右Ctrl)で変更できます。

デフォルトでは、VirtualBoxにフォーカスが当たると自動でONになります。
これを禁止することで、普段はAutoHotKeyが使えるようにします。

2017/12/12追記

ホストキーは右Ctrl+右Shiftにすることにしました。
Windowsのクリップボードツールを使用中に干渉してキャプチャーがONになってしまうからです。

キー配置なしにすると、なぜかAlt+Tabで検索ボックスが出現するので諦めました。

日本語入力のON/OFFショートカットキーを変更する

AutoHotKeyではsetIME()関数を使用して直接日本語入力モードを制御しています。
しかし、ゲストマシンのGoogle日本語入力ではそうはいきません。

そこで、ゲストマシンがアクティブの場合はsetIME()ではなく、日本語入力をON/OFFするキーを割り当てるようにしました。

キーボードの設定を開きます。

最も他のツールと競合しなかった ScrollLockCtrl+ScrollLock を割り当てました。

Pauseキーも含めて色々と試しましたが、AutoHotKeyとの相性もあってかなり苦戦しました。。
AutoHotKeyでは{sc046}がScreenLockキーに該当しますので、以下の様な関数を組みました。

;【概要】日本語入力をON/OFFを指定して切り替える
;【引数】true: 日本語入力ON / false: 日本語入力OFF
;【戻値】なし
setIME(imeOn) {
    if (isActive("ubuntu")) {
        if (imeOn) {
            send {sc046}
        } else {
            send ^{sc046}
        }
    } else {
        IME_SET(imeOn)
    }
}

isActiveIME_SETは上記以外で定義された関数です。

ターミナルの設定

私の場合は以下を設定します。不要の場合はスキップしてください。
端末を開き、ファイル > 新しいプロファイルからプロファイルを作成します。

デフォルトシェルをfishにする

コマンドタブの設定を以下の様にします。

Powerline対応フォントに変更する

全般タブの設定を以下の様にします。

Powerlineフォントはインストールが必要なため、Ansibleのplaybookにtaskを記載しています。
冪等性は考慮していません。

- name: "[Powerline] Clone font"
  git:
    repo: https://github.com/powerline/fonts.git
    dest: /tmp/fonts
    depth: 1

- name: "[Powerline] Install"
  command: ./install.sh
  args:
    chdir: /tmp/fonts

デフォルトプロファイルの変更

ターミナルを開いたときのプロファイルを今回作成したものにしましょう。
端末 > 設定から設定できます。

トラブルシューティング

ウィンドウの切り替えができない

Hostキーを押してからAlt + TABで切り替えできます。
上述したキーボードのキャプチャが有効になっているかを気にする必要があります。

Vimやnano以外のエディタを使いたい

私はVisual Studio Codeを入れています。

Windowsでも使っていますが、以下をコピーすれば使い勝手はほぼ同じでした。

  • settings.json (ユーザ設定)
  • keybindings.json (ユーザのキーショートカット)

画面の動きがもっさりする

Vagrantfileの設定が適切であるかを確認してください。
特に以下のケースでは悪化が著しいです。

  • 仮想マシンのモニタ数が2以上になっている ("--monitorcount", "1"でない)
  • 3DアクセラレータをONになっていない ("--accelerate3d", "on"でない)

VS Codeをコマンドから起動できない

codeで起動できない場合、環境変数DISPLAYの値を確認してください。

  • DISPLAY:0と設定されていればOK
  • DISPLAYlocalhost:0.0だとNG

Vimのヤンクとクリップボードを同期させたい

Ubuntu Desktop

.vimrcset clipboard=unnamedplusを設定しましょう。

set clipboard=unamedでは動作しません。

Windows

.vimrcset clipboard=unnamedplusを設定した上で以下の準備が必要です。

  • Windows Xサーバ(VcXsrv など)のインストール/起動が完了している
  • 環境変数DISPLAYの値が:0である
  • vimで:echo has('clipboard')した結果が1でなければvim-gtk をインストールしている

クリップボードと通信するためWindows Xサーバが必要なのが少し面倒ですね。

tmuxのコピー領域とクリップボードを同期させたい

.tmux.confでバインドを設定しましょう。

Ubuntu Desktop

デフォルトでインストールされているxselを使ってクリップボードを操作させます。

$ bind-key -T copy-mode-vi y send -X copy-pipe-and-cancel "xsel -ip && xsel -op | xsel -ib"

Windows

Windowsコマンドのwcmd clipを使用します。

$ bind-key -T copy-mode-vi y send -X copy-pipe-and-cancel "wcmd clip"

前提として、WSL terminalでwcmdコマンドを使える必要があります。

たまにアプリケーションの立ち上げやクリップボードの同期ができない

VagrantでUbuntuを立ち上げた後、以下のいずれにも該当しないことを確認してください。

  • ゲストセッションでログインしてから、Ubuntu内でvagrantユーザとして再ログインした
  • Ubuntuにログインしていない

アプリケーションの立ち上げやクリップボードの同期を行うには、ファーストログイン時にvagrantユーザとしてログイン済みである必要があります。

webpack-dev-serverのエンドポイントにアクセスできない

hostコマンドを使って、ホストからのアクセスを許可するようにしましょう。 webpack-dev-server --host 0.0.0.0 とすればアクセスできるようになるはずです。

また、変更点を検知するために--watch-poolオプションも必要です。

Hugoのエンドポイントにアクセスできない

bindコマンドを使って、ホストからのアクセスを許可するようにしましょう。 hugo serve --bind 0.0.0.0とすればアクセスできるようになるはずです。

共有ディレクトリを使っている場合Hot Reloadされませんでした。

使用感

Ubuntu Desktopにして良かったこと

Desktop版ならではのメリットは3つです。

  • 純粋なLinuxで動かしたいモノ(Docker等)について
    • 動作を直接ブラウザで確認できる
      • Vagrantのポートフォワーディングはなぜか上手くいかないことがある…
    • Vim以外のエディタで閲覧/編集できる
  • WSL terminalからUbuntuにsshログインしてコマンドを打つと、ゲストの仮想モニタで起動できる
    • firefox blog.mamansoft.netとすればこのブログが立ち上がる
  • クリップボードやyankを完全に同期できる
    • Ubuntuがクリップボードを持ち、かつWindowsと同期できるのが大きい

以下はDesktop版ではないUbuntu環境でも同じです。

  • ターミナルの反応速度が速い
  • Windows特有の落とし穴にハマるリスクを回避できる

Ubuntu Desktopの期待外れだったこと

UbuntuというよりVirtualBoxを使うことによる弊害がほとんだと思いますが…

  • 仮想スクリーンを2つ以上にすると描画が遅い
  • ウィンドウの操作が思考の速度でできない
  • ホストマシンとキーバインディングが競合して悩まされることがある

総括

Ubuntu Desktopで開発できる環境を整えてみました。
現時点では1画面で多少のキー競合を許容できれば快適といったレベルです。

キーバインディングの競合は仮想マシンを使う以上、避けられない問題です。
ウィンドウと複数仮想スクリーンでの挙動は、VirtualBoxに頑張っていただきたいと思いました。

1画面という制限付きでも、GUIが使えるUbuntuから受ける恩恵は大きいです。
基本はWSLターミナルで操作し、画面が必要なときに限って1画面限定のVirtualBox上で操作する… という使い方をしばらく試してみます。

参考