忘備録

日々の調べ物をまとめる。アウトプットする。基本自分用。

Git Flow

Gitの構成管理・運用方法として知られているGit Flowについて学びたいと思います

参考

git-model

環境

  • OS: OS X 10.10.5
  • git version 2.5.4 (Apple Git-61)

準備

git flowをサポートしてくれるツール

その名も「git−flow」(直球)をインストールします

$ brew install git-flow
==> Downloading https://homebrew.bintray.com/bottles/git-flow-0.4.1.yosemite.bot
######################################################################## 100.0%
==> Pouring git-flow-0.4.1.yosemite.bottle.tar.gz
==> Caveats
Bash completion has been installed to:
  /usr/local/etc/bash_completion.d

zsh completion has been installed to:
  /usr/local/share/zsh/site-functions
==> Summary
🍺  /usr/local/Cellar/git-flow/0.4.1: 15 files, 106.6K
# 確認
$ git flow version
0.4.1

git-flow初期化

git-flowを使って初期化を行います

# 開発対象のリポジトリをclone
$ git clone https://github.com/mktktmr/ex_git-flow.git
Cloning into 'ex_git-flow'...
remote: Counting objects: 6, done.
remote: Compressing objects: 100% (3/3), done.
remote: Total 6 (delta 0), reused 0 (delta 0), pack-reused 0
Unpacking objects: 100% (6/6), done.
Checking connectivity... done.
$ cd ex_git-flow/
# 「git flow init」でgit flowに必要な初期化を行う
# オプション-d は初期設定
$ git flow init -d
Using default branch names.

Which branch should be used for bringing forth production releases?
   - master
Branch name for production releases: [master] 
Branch name for "next release" development: [develop] 
$ 
$ git branch -a
* develop
  master
  remotes/origin/HEAD -> origin/master
  remotes/origin/master
$ 
$ git push -u origin develop
Total 0 (delta 0), reused 0 (delta 0)
To https://github.com/mktktmr/ex_git-flow.git
 * [new branch]      develop -> develop
Branch develop set up to track remote branch develop from origin. 

git flowにおけるブランチの役割

master

  • リリース用
  • プロダクトのバージョンでtagを切る
  • 各開発者はこのブランチに対してcommitすることはない

develop

  • 開発用
  • このブランチに開発中の最新ソースがあるよう管理する
  • featureブランチ(後述)をきる元のブランチとなる
  • 各開発者はこのブランチに対してcommitすることはない

feature

  • 作業用
  • 各開発者はこのブランチにcommitをし、作業が終わったらfeatureにPullRequestを送る

release

  • バージョン番号の付与などリリースに関する作業を行う
  • リリース作業中に発見されたバグはこのブランチにコミットしてよい
  • 仕様変更や機能変更は行わない
  • releaseブランチでの作業が完了したらmasterブランチとdevelopブランチにmergeし、masterブランチではタグを切る

hotfix

  • hotfixブランチはフローにおいて必須ではなく、以下のようなことが発生した場合、適宜作成する
    • release後バグが見つかり、かつdevelopブランチのソースがmergeできる状態でない
    • バグや障害の対応が早急に必要で次のリリースバージョンを待つことができない

ワークフロー

featureブランチでの作業

# ブランチを最新化(1つのブランチに対して複数の開発者がいる場合)
$ git pull
Already up-to-date.
$ 
# git-flowでfeatureブランチ追加
$ git flow feature start add-user
Switched to a new branch 'feature/add-user'

Summary of actions:
- A new branch 'feature/add-user' was created, based on 'develop'
- You are now on branch 'feature/add-user'

Now, start committing on your feature. When done, use:

     git flow feature finish add-user

$ 
# 確認
$ git branch -a
  develop
* feature/add-user
  master
  remotes/origin/HEAD -> origin/master
  remotes/origin/develop
  remotes/origin/master
$ 
# featureブランチをリモートにpush
$ git push origin feature/add-user
Total 0 (delta 0), reused 0 (delta 0)
To https://github.com/mktktmr/ex_git-flow.git
 * [new branch]      feature/add-user -> feature/add-user
$
# 実装(した体で)
$ touch AddUser.java
$ 
$ git add .
$ git commit -m "Add AddUser.java"
[feature/add-user aae3200] Add AddUser.java
 1 file changed, 0 insertions(+), 0 deletions(-)
 create mode 100644 AddUser.java
$ 
# pushの前にブランチを最新化
$ git pull origin feature/add-user
From https://github.com/mktktmr/ex_git-flow
 * branch            feature/add-user -> FETCH_HEAD
Already up-to-date.
$ 
# pushの前にdevelopブランチとmergeしておく(他の開発者が更新してるかもしれない)
$ git merge --no-ff origin develop
Already up-to-date.
$ 
# push
$ git push origin feature/add-user
Counting objects: 3, done.
Delta compression using up to 4 threads.
Compressing objects: 100% (2/2), done.
Writing objects: 100% (3/3), 299 bytes | 0 bytes/s, done.
Total 3 (delta 0), reused 0 (delta 0)
To https://github.com/mktktmr/ex_git-flow.git
   114038f..aae3200  feature/add-user -> feature/add-user 
# ここまできたらPull Requestを送る

デフォルトブランチについて

デフォルトブランチとは

GitHubにてリポジトリを選択した際に、最初に選択されるブランチ。

git flow ではdevelopブランチを起点にして作業をするため、git flowを採用するのなら 作業ミスを防ぐためデフォルトブランチをdevelopにすることが望ましい

デフォルトブランチの設定方法

デフォルトブランチの設定を変更したいリポジトリ開き、「setting」タブを選択

f:id:mktktmr:20160212195928p:plain

「Branches」を選択し、Default branchの項目のプルダウンからデフォルトにしたいブランチを選び、「Update」をクリックする

f:id:mktktmr:20160212142146p:plain

Pull Request後のフロー

  1. PRした内容をレビューをしてもらい、フィードバックをもらう
    • テストが通ってない
    • コーディング規約に反している
    • コードが汚い
    • リファクタリングの余地がある
    • etc...
  2. レビューのフィードバックを反映させ、pushする
  3. 問題がなくなるまで、1〜2を繰り返す
  4. 管理者がdevelpブランチにマージする

Pull RequestからPR mergeまでの例

pull requestをする

f:id:mktktmr:20160212140429p:plain

フィードバックがある。。。

f:id:mktktmr:20160212140740p:plain

# フィードバックをもとに修正
$ vi AddUser.java 
$ git add .
$ git commit -m "Done Refactoring"
[feature/add-user b35e838] Done Refactoring
 1 file changed, 5 insertions(+)
$ 
# 修正内容をpush
$ git push origin feature/add-user
Counting objects: 3, done.
Delta compression using up to 4 threads.
Compressing objects: 100% (3/3), done.
Writing objects: 100% (3/3), 365 bytes | 0 bytes/s, done.
Total 3 (delta 0), reused 0 (delta 0)
To https://github.com/mktktmr/ex_git-flow.git
   aae3200..b35e838  feature/add-user -> feature/add-user

フィードバックが反映され、問題がなければ、pull requestがマージされる

f:id:mktktmr:20160212141344p:plain

# pull request がマージされたらローカルのdevelpブランチを更新する
$ git checkout develop
Switched to branch 'develop'
Your branch is up-to-date with 'origin/develop'.
$ git pull
remote: Counting objects: 1, done.
remote: Total 1 (delta 0), reused 0 (delta 0), pack-reused 0
Unpacking objects: 100% (1/1), done.
From https://github.com/mktktmr/ex_git-flow
   1e91db3..5b91113  develop    -> origin/develop
Updating 1e91db3..5b91113
Fast-forward
 AddUser.java | 6 ++++++
 1 file changed, 6 insertions(+)
 create mode 100644 AddUser.java

以上を繰り返し機能を追加していく。

featureブランチを切る粒度については今後の研究課題。。。

releaseブランチでの作業

featureブランチによる機能開発が全て終わり、マージが済んだらリリース作業用としてreleaseブランチを切る

このブランチで行うことはバグの修正のみであり、機能追加などはしてはいけない

# ブランチの確認
$ git branch
* develop
  feature/add-user
  master
$ 
# 最新の状態にしておく
$ git pull
Already up-to-date.
# git-flowでreleaseブランチを作成する
$ git flow release start "1.0.0"
Switched to a new branch 'release/1.0.0'

Summary of actions:
- A new branch 'release/1.0.0' was created, based on 'develop'
- You are now on branch 'release/1.0.0'

Follow-up actions:
- Bump the version number now!
- Start committing last-minute fixes in preparing your release
- When done, run:

     git flow release finish '1.0.0'

$ 
# ブランチ確認
$ git branch
  develop
  feature/add-user
  master
* release/1.0.0
$ 
# バグが見つかったので、修正する(という体)
$ vi AddUser.java 
$ git commit -am "Bugfix"
[release/1.0.0 8ea7af0] Bugfix
 1 file changed, 1 insertion(+)
$ 
# git-flowでrelease 作業のクローズする
# git-flow release finishを実行すると以下の処理が走る
# 1. releaseブランチがmasterブランチにマージされる
# 2. タグの発行
# 3. releaseブランチがdevelopブランチにマージされる
$ git flow release finish
Missing argument <version>
usage: git flow release [list] [-v]
       git flow release start [-F] <version>
       git flow release finish [-Fsumpk] <version>
       git flow release publish <name>
       git flow release track <name>
$ git flow release finish "1.0.0"
Switched to branch 'master'
Your branch is up-to-date with 'origin/master'.
Merge made by the 'recursive' strategy.
 AddUser.java | 7 +++++++
 1 file changed, 7 insertions(+)
 create mode 100644 AddUser.java
Switched to branch 'develop'
Your branch is up-to-date with 'origin/develop'.
Merge made by the 'recursive' strategy.
 AddUser.java | 1 +
 1 file changed, 1 insertion(+)
Deleted branch release/1.0.0 (was 8ea7af0).

Summary of actions:
- Latest objects have been fetched from 'origin'
- Release branch has been merged into 'master'
- The release was tagged '1.0.0'
- Release branch has been back-merged into 'develop'
- Release branch 'release/1.0.0' has been deleted

$ 
# タグの確認
$ git tag
1.0.0
$ 
# ブランチ確認
$ git branch
* develop
  feature/add-user
  master
$
# developブランチのpush
$ git push origin develop
Counting objects: 4, done.
Delta compression using up to 4 threads.
Compressing objects: 100% (4/4), done.
Writing objects: 100% (4/4), 434 bytes | 0 bytes/s, done.
Total 4 (delta 2), reused 0 (delta 0)
To https://github.com/mktktmr/ex_git-flow.git
   5b91113..8f133fb  develop -> develop
$ 
# masterブランチのpush
$ git checkout master
Switched to branch 'master'
Your branch is ahead of 'origin/master' by 5 commits.
  (use "git push" to publish your local commits)
$ git push
Counting objects: 1, done.
Writing objects: 100% (1/1), 241 bytes | 0 bytes/s, done.
Total 1 (delta 0), reused 0 (delta 0)
To https://github.com/mktktmr/ex_git-flow.git
   1e91db3..5efe88e  master -> master
$ 
# タグをpush
$ git push --tag
Counting objects: 1, done.
Writing objects: 100% (1/1), 175 bytes | 0 bytes/s, done.
Total 1 (delta 0), reused 0 (delta 0)
To https://github.com/mktktmr/ex_git-flow.git
 * [new tag]         1.0.0 -> 1.0.0

hotfixブランチでの作業

# 1.0.0タグからhotfix/1.0.1ブランチを作成
$ git flow hotfix start '1.0.1' '1.0.0'
Switched to a new branch 'hotfix/1.0.1'

Summary of actions:
- A new branch 'hotfix/1.0.1' was created, based on '1.0.0'
- You are now on branch 'hotfix/1.0.1'

Follow-up actions:
- Bump the version number now!
- Start committing your hot fixes
- When done, run:

     git flow hotfix finish '1.0.1'

# バグを修正する(体で)
$ vi AddUser.java 
# 修正をコミット
$ git commit -am "HotFix"
[hotfix/1.0.1 4df2872] HotFix
 1 file changed, 1 insertion(+)
$ 
$ git push origin 
$ git branch
  develop
  feature/add-user
* hotfix/1.0.1
  master
$ git push origin hotfix/1.0.1
Counting objects: 3, done.
Delta compression using up to 4 threads.
Compressing objects: 100% (3/3), done.
Writing objects: 100% (3/3), 331 bytes | 0 bytes/s, done.
Total 3 (delta 1), reused 0 (delta 0)
To https://github.com/mktktmr/ex_git-flow.git
 * [new branch]      hotfix/1.0.1 -> hotfix/1.0.1

バグを修正し、hotfixブランチをpushしたらPRを送る

merge対象はmasterブランチとなることがポイント

f:id:mktktmr:20160215203322p:plain

特に問題もなく、PRが取り込まれたとする

f:id:mktktmr:20160215203649p:plain

タグの作成とリリース

f:id:mktktmr:20160215211739p:plain

f:id:mktktmr:20160215211749p:plain

f:id:mktktmr:20160215211821p:plain

f:id:mktktmr:20160215213627p:plain

タグを切ったら、hotfixとしての対応は終了だが、developブランチと差分が出て入るはずのなので、developブランチにもPRを送り、適宜mergeする

# ブランチを最新化
$ git pull
remote: Counting objects: 1, done.
remote: Total 1 (delta 0), reused 0 (delta 0), pack-reused 0
Unpacking objects: 100% (1/1), done.
From https://github.com/mktktmr/ex_git-flow
   8f133fb..f04db69  develop    -> origin/develop
 * [new tag]         1.0.1      -> 1.0.1
Already up-to-date.
# タグの確認
$ git tag
1.0.0
1.0.1
$ git checkout develop
Switched to branch 'develop'
Your branch is behind 'origin/develop' by 3 commits, and can be fast-forwarded.
  (use "git pull" to update your local branch)
# ブランチの最新化
$ git pull origin develop
From https://github.com/mktktmr/ex_git-flow
 * branch            develop    -> FETCH_HEAD
Updating 8f133fb..f04db69
Fast-forward
 AddUser.java | 1 +
 1 file changed, 1 insertion(+)
# タグの確認
$ git tag
1.0.0
1.0.1 

【Linux】セキュアなサーバを構築するための最低限の設定

サーバ立て直すことにしたのでメモ

環境

CentOS 6.7

前提

OSのインストールは最小構成(minimal)で行っているものとする

参考

Linuxサーバーセキュリティ徹底入門 オープンソースによるサーバー防衛の基本

作業用ユーザの作成

rootユーザでの作業はしないようにするため、作業用のユーザを作成

# ユーザ作成
$ adduser hoge
# パスワード設定
$ passwd hoge
Changing password for user hoge.
New password:
Retype:
password: all authentication tokens updated successfully.

システムのアップデート

システムの状態を最新にする

$ yum -y update
読み込んだプラグイン:fastestmirror
更新処理の設定をしています
Loading mirror speeds from cached hostfile
 * base: ftp.riken.jp
 * extras: ftp.riken.jp
 * updates: ftp.riken.jp
依存性の解決をしています
#〜〜〜〜〜〜〜(中略)〜〜〜〜〜〜〜〜〜
  selinux-policy-targeted.noarch 0:3.7.19-279.el6_7.8             sqlite.x86_64 0:3.6.20-1.el6_7.2                          
  sudo.x86_64 0:1.8.6p3-20.el6_7                                  tzdata.noarch 0:2016a-2.el6                               
  udev.x86_64 0:147-2.63.el6_7.1                                 

完了しました!

不要なサービスの停止

動作しているサービスの確認

$ chkconfig --list | grep 3:on
auditd          0:off  1:off  2:on   3:on   4:on   5:on   6:off
blk-availability    0:off  1:on   2:on   3:on   4:on   5:on   6:off
crond           0:off  1:off  2:on   3:on   4:on   5:on   6:off
ip6tables       0:off  1:off  2:on   3:on   4:on   5:on   6:off
iptables        0:off  1:off  2:on   3:on   4:on   5:on   6:off
iscsi           0:off  1:off  2:off  3:on   4:on   5:on   6:off
iscsid          0:off  1:off  2:off  3:on   4:on   5:on   6:off
lvm2-monitor    0:off  1:on   2:on   3:on   4:on   5:on   6:off
mdmonitor       0:off  1:off  2:on   3:on   4:on   5:on   6:off
netfs           0:off  1:off  2:off  3:on   4:on   5:on   6:off
network         0:off  1:off  2:on   3:on   4:on   5:on   6:off
postfix         0:off  1:off  2:on   3:on   4:on   5:on   6:off
rsyslog         0:off  1:off  2:on   3:on   4:on   5:on   6:off
sshd            0:off  1:off  2:on   3:on   4:on   5:on   6:off
udev-post       0:off  1:on   2:on   3:on   4:on   5:on   6:off

不要なサービスの停止

$ service ip6tables stop
ip6tables: チェインをポリシー ACCEPT に設定中: filter      [  OK  ]
ip6tables: ファイアウォールルールを消去中:                 [  OK  ]
ip6tables: モジュールを取り外し中:                         [  OK  ]
$ service netfs stop
$ service postfix stop
postfix を停止中:                                          [  OK  ]

ログインの制限

sshから直接rootユーザにログインできなくする

以下のファイルを編集
/etc/ssh/sshd_config

#PermitRootLogin yes

PermitRootLogin no

に変更

sshのポートを変更

ウェルノウンポートのままだと攻撃を受けやすいため変更しておく

以下のファイルを編集
/etc/ssh/sshd_config

Port 22

Port 20022

に変更

併せてiptablesの設定も変更しておく

以下のファイルを変更
/etc/sysconfig/iptables

-A INPUT -m state --state NEW -m tcp -p tcp --dport 22 -j ACCEPT

-A INPUT -m state --state NEW -m tcp -p tcp --dport 20022 -j ACCEPT

に変更

sshiptablesを再起動する

$ service sshd restart
$ service iptables restart

ネットワークの確認

余計なポートが開いていないか確認

$ netstat -atun
Active Internet connections (servers and established)
Proto Recv-Q Send-Q Local Address               Foreign Address             State      
tcp        0      0 0.0.0.0:20022               0.0.0.0:*                   LISTEN      
tcp        0     48 192.168.100.251:20022       192.168.100.100:64595       ESTABLISHED 
tcp        0      0 :::20022                    :::*                        LISTEN      
udp        0      0 0.0.0.0:68                  0.0.0.0:*                        

ファイアウォールの設定を確認

Chain INPUT (policy ACCEPT)
target     prot opt source               destination         
ACCEPT     all  --  anywhere             anywhere            state RELATED,ESTABLISHED 
ACCEPT     icmp --  anywhere             anywhere            
ACCEPT     all  --  anywhere             anywhere            
ACCEPT     tcp  --  anywhere             anywhere            state NEW tcp dpt:20022 
REJECT     all  --  anywhere             anywhere            reject-with icmp-host-prohibited 

Chain FORWARD (policy ACCEPT)
target     prot opt source               destination         
REJECT     all  --  anywhere             anywhere            reject-with icmp-host-prohibited 

Chain OUTPUT (policy ACCEPT)
target     prot opt source               destination

前回立てたサーバーはこれくらいのことしかしてなかった。

今回はもう少し突っ込んで、セキュリティ対策してみようと思う(するとは言っていない)

【CentOS6】インストール後に設定しておきたいこと

今更感溢れるCentOS6系の設定について

環境

前提

  • OSを最小構成にてインストールしていること

ネットワークインターフェースの自動起動

デフォルトの設定ではネットワークインターフェースが自動的に起動しないようになっている

自動起動するためには以下のファイルを書き換える

/etc/sysconfig/network-scripts/ifcfg-eth0

ONBOOT=yes # noをyesに変更

キーボード

インストール時ににキーボードの設定を誤ってしまったなど、キーボードの設定を変更したい場合、以下のファイルを書き換えると変更できる

/etc/sysconfig/keyboard

例)日本語配列にしたい場合の設定

KEYTABLE="jp106"
MODEL="jp106"
LAYOUT="jp"
KEYBOARDTYPE="pc"

参考:CentOSでキーボード設定が日本語じゃないときの対処法: m6 BLOG

システム時刻

# ntpdateのインストール
$ yum install ntpdate
読み込んだプラグイン:fastestmirror
インストール処理の設定をしています
#〜〜〜〜〜〜〜〜〜〜〜中略〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜
インストール:
  ntpdate.x86_64 0:4.2.6p5-5.el6.centos.4                                       

完了しました!
# システム時刻の設定
$ ntpdate ntp.nict.jp
15 Feb 03:18:47 ntpdate[2539]: step time server 133.243.238.244 offset 471.270285 sec
# 確認
$ date
2016215日 月曜日 03:18:50 JST
# ついでにハードウェアクロックも修正
# システムクロックの時刻をハードウェアクロックに反映
$ hwclock -w
# 確認
$ hwclock -r
20160215031928秒  -0.609921 秒

cronで定期実行する場合(下記の例は毎月1日の0:00に実行)

0 0 1 * * /usr/sbin/ntpdate ntp.nict.jp > /dev/null 2>&1
0 0 1 * * /sbin/hwclock -w > /dev/null 2>&1

参考:LPIC Linux - システム時刻の維持

MeteorやっててClient IDの取得でハマった

環境

Meteor: 1.2.1

前提

Googleアカウントを持っていること

accounts-googleの実装

MeteorでGoogleのOAuthによるログインを実装するのはすごく簡単で、
以下の通りパッケージを追加して、コードを一行差し込むだけでできてしまう。

パッケージ追加

meteor add accounts-google accounts-ui

sample.html

<head>
  <title>test-meteor-pja</title>
</head>

<body>
  <h1>Welcome to Meteor!</h1>

  {{> loginButtons}}  <!-- add -->

  {{> hello}}
</body>

<template name="hello">
  <button>Click Me</button>
  <p>You've pressed the button {{counter}} times.</p>
</template>

アクセスするとご覧の通り

f:id:mktktmr:20160212231237p:plain

ただ、client IDの登録でハマります。

私のようなイングリッシュがプアな人間は特に。

ログインボタンを実装後、クリックするとClient IDの登録手順が示されるんですが、これが古いもののためのなのか、書いてあることと実際の画面が全然異なるのである。

f:id:mktktmr:20160212085747p:plain

というわけで、OAuth認証ができるようになるまでを説明していく。

1. Google Developers Consoleへアクセス

まずは「Google Developers Console」にアクセスする

https://console.developers.google.com/

f:id:mktktmr:20160212074024p:plain

2. プロジェクトの作成

はじめてGoogle Developers Consoleを利用する人はプロジェクトがないはずなので「Create project」からプロジェクトを作成する

f:id:mktktmr:20160210135939p:plain

f:id:mktktmr:20160210135942p:plain

3. OAuth consentの設定

プロジェクトが作成できたら、ダッシュボードがひらくので、 左上にあるハンバーガーアイコンをクリックし、メニューを開く

f:id:mktktmr:20160210135945p:plain

メニューのなかに「API Manager」という項目があるので選択する

f:id:mktktmr:20160210135949p:plain

API Managerページのメニューのなかに「Credentials」があるので選択し、開かれたページのタブのなかの「OAuth consent screen」を選択する f:id:mktktmr:20160210135953p:plain

メールアドレスとProduct nameを入力し、「save」ボタンをクリック

f:id:mktktmr:20160212234056p:plain

4. Credentialの作成

次に「Credentials」タブを選択し、「New Credentials」ボタンを押下する

f:id:mktktmr:20160210140000p:plain

すると、プルダウンが開き、そのなかに「OAuth Client ID」があるので選択する。

f:id:mktktmr:20160210140004p:plain

  1. Application type
    Web applicationを選択

  2. Name
    任意の名前を入力

  3. Authorized JavaScript origins
    http://localhost:3000

  4. Authorized redirect URIs
    http://localhost:3000/_oauth/google

以上を入力し、「Create」ボタンをクリック

f:id:mktktmr:20160212232948p:plain

すると「client ID」と「client secret」が発行される

f:id:mktktmr:20160210140031p:plain

5. 「client ID」および「client secret」の登録

手順4で発行された「client ID」および「client secret」をMeteorアプリケーションのOAuth設定のフォームに入力し、「Save Configration」をクリックする

f:id:mktktmr:20160212085252p:plain

↓のようになればおk

f:id:mktktmr:20160212233625p:plain

【React】とりあえず環境

Reactを弄る環境を整えてみる

環境

  • OS: OS X 10.10.5
  • React: 0.14.7
  • npm: 3.3.12

前提

  • npmがインストールされていること

参考

Getting Started | React

npmを使った方法

CommonJSのモジュールシステムとして「browserify」と「webpack」を使う方法があるが、今回はbrowserifyを利用する

※「browserify」とは

browserify をはじめてみる - Please Sleep

CommonJS のモジュールの仕組み、つまり Node.js の require をブラウザ上でも使えるようにするもの、ということでいいみたい。

公式:Browserify

browserifyのインストール

$ sudo npm install -g browserify
Password:
/usr/local/bin/browserify -> /usr/local/lib/node_modules/browserify/bin/cmd.js
/usr/local/lib
└─┬ browserify@13.0.0 
  ├── assert@1.3.0 
#〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜中略〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜
  ├─┬ vm-browserify@0.0.4 
  │ └── indexof@0.0.1 
  └── xtend@4.0.1 

Reactのインストール

$ npm install --save react react-dom babelify babel-preset-react
npm WARN saveError ENOENT: no such file or directory, open '/path/package.json'
/path/Using_React_from_npm
├─┬ babel-preset-react@6.3.13 
│ ├─┬ babel-plugin-syntax-flow@6.3.13 
#〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜中略〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜〜
│   └── whatwg-fetch@0.9.0 
└── react-dom@0.14.7 

npm WARN ENOENT ENOENT: no such file or directory, open '/path/Using_React_from_npm/package.json'
npm WARN EPACKAGEJSON Using_React_from_npm No description
npm WARN EPACKAGEJSON Using_React_from_npm No repository field.
npm WARN EPACKAGEJSON Using_React_from_npm No README data
npm WARN EPACKAGEJSON Using_React_from_npm No license field.

ワーニングは後で調べます。。。

サンプルソース

サンプルソースとして以下を用意する

helloworld.html

<!DOCTYPE html>
<html>
  <head>
    <meta charset="UTF-8" />
    <title>Hello React!</title>
  </head>
  <body>
    <div id="example"></div>
    <script src="bundle.js"></script>
  </body>
</html>

main.js

var React = require('react');
var ReactDOM = require('react-dom');

ReactDOM.render(
  <h1>Hello, world!</h1>,
  document.getElementById('example')
);

トランスパイル

main.jsをbundle.jsとしてトランスパイルする

$ browserify -t [ babelify --presets [ react ] ] main.js -o bundle.js

HTMLファイルを開いてみると

f:id:mktktmr:20160203160602p:plain

ちなみに最終的なディレクトリ構成以下です

Root/
├ bundle.js
├ helloworld.html
├ main.js
└ node_modules/

Error

ちょっとハマりました。。。

当初HTMLを以下のように書いていて

<!DOCTYPE html>
<html>
  <head>
    <meta charset="UTF-8" />
    <title>Hello React!</title>
  </head>
  <body>
    <div id="example"></div>
    <script src="bundle.js"></script>
  </body>
</html>

↓こんなエラー出てました。。。

f:id:mktktmr:20160203161011p:plain

Invariant Violation: _registerComponent(...): Target container is not a DOM element.
invariantbundle.js:908
_registerComponentbundle.js:13104:165
_renderNewRootComponentbundle.js:13126
ReactMount__renderNewRootComponentbundle.js:14476
_renderSubtreeIntoContainerbundle.js:13206
renderbundle.js:13226
React_renderbundle.js:14476
(anonymous 関数)bundle.js:34
sbundle.js:16
ebundle.js:25
(anonymous 関数)bundle.js:27

javascript - Invariant Violation: _registerComponent(...): Target container is not a DOM element - Stack Overflow

上記のコードだと、スクリプトが読まれた時点では、バインドしたいDOMが生成されていなため、エラーをこいてしまうってことだと思う。

スクリプトを読み込む位置を変えずに、

<script async src="bundle.js"></script> <!-- add async -->

でもいけるが、スクリプトを読み込む位置を変える方がベターらしい

starter kitを使った方法

starter kitを利用するとnpmなしでもReactをできるよ!とのことでこちらも試してみる

starter kitのダウンロード

f:id:mktktmr:20160215171515p:plain

とりあえずHTMLに直書きでReactのコードを差し込む

以下HTMLファイルをダウンロードしてきたディレクトリ配下に置く

react-0.14.7/
┣ README.md
┣ build
┣ examples
helloworld.html

helloworld.html

<!DOCTYPE html>
<html>
  <head>
    <meta charset="UTF-8" />
    <title>Hello React!</title>
    <script src="build/react.js"></script>
    <script src="build/react-dom.js"></script>
    <script src="https://cdnjs.cloudflare.com/ajax/libs/babel-core/5.8.23/browser.min.js"></script>
  </head>
  <body>
    <div id="example"></div>
    <script type="text/babel">
      ReactDOM.render(
        <h1>Hello, world!</h1>,
        document.getElementById('example')
      );
    </script>
  </body>
</html>

helloworld.htmlを開いてみる

f:id:mktktmr:20160215173414p:plain

HTMLとjsファイルを分割してみる

以下jsファイルを追加

react-0.14.7/
┣ README.md
┣ build
┣ examples
┣ helloworld.html
┗ src/
    ┗ helloworld.js

ReactDOM.render(
  <h1>Hello, world!!</h1>,
  document.getElementById('example')
);

上記jsファイルを読み込むようHTMLを修正

<!DOCTYPE html>
<html>
  <head>
    <meta charset="UTF-8" />
    <title>Hello React!</title>
    <script src="build/react.js"></script>
    <script src="build/react-dom.js"></script>
    <script src="https://cdnjs.cloudflare.com/ajax/libs/babel-core/5.8.23/browser.min.js"></script>
  </head>
  <body>
    <div id="example"></div>
    <script type="text/babel" src="src/helloworld.js"></script> <!-- 修正 -->
  </body>
</html>

helloworld.htmlを開く

f:id:mktktmr:20160215175725p:plain

※ちなみに公式の説明にもある通りブラウザによっては直接HTMLファイルを開くとエラーが発生することがある

Note that some browsers (Chrome, e.g.) will fail to load the file unless it's served via HTTP.

Chromeで開くとこんな具合

f:id:mktktmr:20160215180504p:plain

そんな時はpythonのhttpモジュールを呼ぶと便利(最近知った)

# python -m SimpleHTTPServerと一行叩くだけでhttpseverが起動する。。。
$ python -m SimpleHTTPServer
Serving HTTP on 0.0.0.0 port 8000 ...

んで、http://localhost:8000/helloworld.htmlにアクセスすると

f:id:mktktmr:20160215180847p:plain

オフラインでトランスパイル

Babelを利用することで、オフラインでもJavaScriptをトランスパイルできる

※「Babel」とは

Babelで始める!モダンJavaScript開発 | HTML5Experts.jp

ECMAScript2015 (ES6)やECMAScript7などで書かれたソースコードを一般的なブラウザがサポートしているECMAScript5の形式に出力することができます。

とりあえずbabelをインストール

npm install --global babel-cli
npm install babel-preset-react

_人人人人人人人人人人人人人_
> Quick Start Without npm <
 ̄Y^Y^Y^Y^Y^Y^Y^YY^Y^Y^ ̄

なんて銘打っておいてnpm使ってるじゃんっていうツッコミをしたくなるんですが、まあ、置いときます

ソースファイル修正

以下jsファイルを修正

src/helloworld.js

ReactDOM.render(
  React.createElement('h1', null, 'Hello, world!!!'),
  document.getElementById('example')
);

HTMLファイルも修正

src/helloworld.html

<!DOCTYPE html>
<html>
  <head>
    <meta charset="UTF-8" />
    <title>Hello React!</title>
    <script src="build/react.js"></script>
    <script src="build/react-dom.js"></script>
    <!-- No need for Babel! -->
  </head>
  <body>
    <div id="example"></div>
    <script src="build/helloworld.js"></script>
  </body>
</html>

bable発動

# srcディレクトリを監視して、トランスパイルしたファイルをbuildディレクトリに出力する
$ babel --presets react src --watch --out-dir build
src/helloworld.js -> build/helloworld.js

ブラウザでhtmlファイルオープン

f:id:mktktmr:20160215185051p:plain

こんな感じで色々な方法でReactを利用できる

どれがベスト・ベターな方法なのかはこれから探っていく

個人的にはReactはWebフロントのJavaScriptフレームワークの中で一番注目している

また、WebだけでなくNativeアプリを向けのReact Nativeなんてのも開発中みたいで、これから先も楽しみなフレームワークです

【Heroku】チュートリアル(Node.js)メモ

環境

前提

  • Heroku Toolbeltがインストールされていること
  • node.jsがインストールされていること

デプロイまで

#公式サンプルプロジェクトを利用
$ git clone https://github.com/heroku/node-js-getting-started.git
Cloning into 'node-js-getting-started'...
remote: Counting objects: 442, done.
remote: Total 442 (delta 0), reused 0 (delta 0), pack-reused 442
Receiving objects: 100% (442/442), 224.68 KiB | 119.00 KiB/s, done.
Resolving deltas: 100% (65/65), done.
Checking connectivity... done.
$ cd node-js-getting-started/
#リモートリポジトリの確認
$ git remote
origin #この時点でリモートリポジトリはoriginのみ
$ 
$ 
#「heroku create」でheroku上にアプリを作成する
#createの後(例だと’test-heroku-app’)はアプリ名で省略可能
#省略した場合、自動でランダムな名称が付けられる
$ heroku create test-heroku-app 
Creating test-heroku-app... !!!
 ▸    Name is already taken
#既に使われてる名前でした。。。
$ 
#リトライ
$ heroku create mktktmr-test-heroku-app
Creating mktktmr-test-heroku-app... done, stack is cedar-14
https://mktktmr-test-heroku-app.herokuapp.com/ | https://git.heroku.com/mktktmr-test-heroku-app.git
#リモートリポジトリ確認
$ git remote
heroku #「heroku create」をするとリモートリポジトリにherokuが追加される
origin

Herokuダッシュボード(heroku create後) f:id:mktktmr:20160130072527p:plain

App管理画面 f:id:mktktmr:20160130072626p:plain

push

#herokuリポジトリにPushすることでデプロイ完了
$ git push heroku master
Counting objects: 433, done.
Delta compression using up to 4 threads.
Compressing objects: 100% (337/337), done.
Writing objects: 100% (433/433), 222.43 KiB | 0 bytes/s, done.
Total 433 (delta 63), reused 433 (delta 63)
remote: Compressing source files... done.
remote: Building source:
remote: 
remote: -----> Node.js app detected
remote: 
remote: -----> Creating runtime environment
remote:        
remote:        NPM_CONFIG_LOGLEVEL=error
remote:        NPM_CONFIG_PRODUCTION=true
remote:        NODE_ENV=production
remote:        NODE_MODULES_CACHE=true
remote: 
remote: -----> Installing binaries
remote:        engines.node (package.json):  0.12.7
remote:        engines.npm (package.json):   unspecified (use default)
remote:        
remote:        Downloading and installing node 0.12.7...
remote:        Using default npm version: 2.11.3
remote: 
remote: -----> Restoring cache
remote:        Skipping cache restore (new runtime signature)
remote: 
remote: -----> Building dependencies
remote:        Pruning any extraneous modules
remote:        Installing node modules (package.json)
remote:        ejs@2.3.3 node_modules/ejs
remote:        
remote:        express@4.13.3 node_modules/express
remote:        ├── escape-html@1.0.2
remote:        ├── merge-descriptors@1.0.0
remote:        ├── array-flatten@1.1.1
remote:        ├── cookie@0.1.3
remote:        ├── utils-merge@1.0.0
remote:        ├── cookie-signature@1.0.6
remote:        ├── fresh@0.3.0
remote:        ├── methods@1.1.2
remote:        ├── range-parser@1.0.3
remote:        ├── vary@1.0.1
remote:        ├── path-to-regexp@0.1.7
remote:        ├── parseurl@1.3.1
remote:        ├── content-type@1.0.1
remote:        ├── etag@1.7.0
remote:        ├── content-disposition@0.5.0
remote:        ├── depd@1.0.1
remote:        ├── qs@4.0.0
remote:        ├── on-finished@2.3.0 (ee-first@1.1.1)
remote:        ├── finalhandler@0.4.0 (unpipe@1.0.0)
remote:        ├── debug@2.2.0 (ms@0.7.1)
remote:        ├── proxy-addr@1.0.10 (forwarded@0.1.0, ipaddr.js@1.0.5)
remote:        ├── send@0.13.0 (destroy@1.0.3, statuses@1.2.1, ms@0.7.1, mime@1.3.4, http-errors@1.3.1)
remote:        ├── serve-static@1.10.2 (escape-html@1.0.3, send@0.13.1)
remote:        ├── type-is@1.6.10 (media-typer@0.3.0, mime-types@2.1.9)
remote:        └── accepts@1.2.13 (negotiator@0.5.3, mime-types@2.1.9)
remote: 
remote: -----> Caching build
remote:        Clearing previous node cache
remote:        Saving 2 cacheDirectories (default):
remote:        - node_modules
remote:        - bower_components (nothing to cache)
remote: 
remote: -----> Build succeeded!
remote:        ├── ejs@2.3.3
remote:        └── express@4.13.3
remote:        
remote: -----> Discovering process types
remote:        Procfile declares types -> web
remote: 
remote: -----> Compressing...
remote:        Done: 10.2M
remote: -----> Launching...
remote:        Released v3
remote:        https://mktktmr-test-heroku-app.herokuapp.com/ deployed to Herok
remote: 
remote: Verifying deploy... done.
To https://git.heroku.com/mktktmr-test-heroku-app.git
 * [new branch]      master -> master

App管理画面(push後)

f:id:mktktmr:20160130074217p:plain

アプリにアクセスしてみる

アプリケーションを開く(Webブラウザが起動し、アプリのURLが開かれる)
$ heroku open
Opening mktktmr-test-heroku-app... done
$

f:id:mktktmr:20160130073517p:plain

その他

ロギング

Logging | Heroku Dev Center

「heroku logs --tail」を利用することでリモートでログ監視ができる

$ heroku logs --tail
2016-01-29T21:22:05.932126+00:00 heroku[api]: Enable Logplex by hoge@fuga.com
2016-01-29T21:22:05.932126+00:00 heroku[api]: Release v2 created by hoge@fuga.com
2016-01-29T21:26:17.536538+00:00 heroku[api]: Scale to web=1 by hoge@fuga.com
2016-01-29T21:26:17.607423+00:00 heroku[api]: Deploy 5e531fb by hoge@fuga.com
2016-01-29T21:26:17.607423+00:00 heroku[api]: Release v3 created by hoge@fuga.com
2016-01-29T21:26:17.921584+00:00 heroku[slug-compiler]: Slug compilation started
2016-01-29T21:26:17.921589+00:00 heroku[slug-compiler]: Slug compilation finished
2016-01-29T21:26:18.689167+00:00 heroku[web.1]: Starting process with command `node index.js`
2016-01-29T21:26:20.729829+00:00 app[web.1]: Node app is running on port 10971
2016-01-29T21:26:22.256715+00:00 heroku[web.1]: State changed from starting to up
2016-01-29T21:47:19.821604+00:00 heroku[router]: at=info method=GET path="/" host=mktktmr-test-heroku-app.herokuapp.com request_id=5c1440f8-8a27-4cfe-9b01-f9da5cbabbef fwd="106.161.123.219" dyno=web.1 connect=1ms service=36ms status=200 bytes=6119

Procfile

Process Types and the Procfile | Heroku Dev Center

アプリケーションが起動した時にすべきコマンドが定義されているファイル

$ ls -l
total 40
-rw-r--r--  1 makoto  staff    19  1 30 06:18 Procfile #こいつ
-rw-r--r--  1 makoto  staff  1371  1 30 06:18 README.md
-rw-r--r--  1 makoto  staff   300  1 30 06:18 app.json
-rw-r--r--  1 makoto  staff   460  1 30 06:18 index.js
-rw-r--r--  1 makoto  staff   486  1 30 06:18 package.json
drwxr-xr-x  5 makoto  staff   170  1 30 06:18 public
drwxr-xr-x  4 makoto  staff   136  1 30 06:18 views
$ 
#中身の確認
$ cat Procfile 
web: node index.js

スケーリング

「heroku ps」でアプリがいくつのdynosで動いているか確認できる

$ heroku ps
=== web (Free): node index.js
web.1: up 2016/01/30 08:19:41 +0900 (~ 3m ago)

「heroku ps:scale」でスケーリング出来る

$ heroku ps:scale web=1
Scaling dynos... done, now running web at 1:Free.
$
#無料アカウントではスケールアップはできないけどもね
$ heroku ps:scale web=2
Scaling dynos... failed
 !    Cannot update to more than 1 Free size dynos per process type.

依存性の定義(Node.jsの場合)

  • 依存性はpackage.jsonに定義する
  • プロジェクトのルートディレクトリにpackage.jsonがあるとHerokuはそのプロジェクトがNode.jsプロジェクトだと認識する
  • ローカルでアプリを実行したい場合、「npm install」で依存ライブラリをインストールする必要がある
$ heroku local web
forego | starting web.1 on port 5000
web.1  | module.js:327
web.1  |     at require (internal/module.js:12:17)

npm install実行

$ npm install
node-js-getting-started@0.1.5 ~/node-js-getting-started
├── ejs@2.3.3 
└─┬ express@4.13.3 
  ├─┬ accepts@1.2.13 
  │ ├─┬ mime-types@2.1.9 
  │ │ └── mime-db@1.21.0 
  │ └── negotiator@0.5.3 
  ├── array-flatten@1.1.1 
  ├── content-disposition@0.5.0 
  ├── content-type@1.0.1 
  ├── cookie@0.1.3 
  ├── cookie-signature@1.0.6 
  ├─┬ debug@2.2.0 
  │ └── ms@0.7.1 
  ├── depd@1.0.1 
  ├── escape-html@1.0.2 
  ├── etag@1.7.0 
  ├─┬ finalhandler@0.4.0 
  │ └── unpipe@1.0.0 
  ├── fresh@0.3.0 
  ├── merge-descriptors@1.0.0 
  ├── methods@1.1.2 
  ├─┬ on-finished@2.3.0 
  │ └── ee-first@1.1.1 
  ├── parseurl@1.3.1 
  ├── path-to-regexp@0.1.7 
  ├─┬ proxy-addr@1.0.10 
  │ ├── forwarded@0.1.0 
  │ └── ipaddr.js@1.0.5 
  ├── qs@4.0.0 
  ├── range-parser@1.0.3 
  ├─┬ send@0.13.0 
  │ ├── destroy@1.0.3 
  │ ├─┬ http-errors@1.3.1 
  │ │ └── inherits@2.0.1 
  │ ├── mime@1.3.4 
  │ └── statuses@1.2.1 
  ├─┬ serve-static@1.10.2 
  │ ├── escape-html@1.0.3 
  │ └─┬ send@0.13.1 
  │   ├── depd@1.1.0 
  │   └── destroy@1.0.4 
  ├─┬ type-is@1.6.10 
  │ └── media-typer@0.3.0 
  ├── utils-merge@1.0.0 
  └── vary@1.0.1

npm install後

$ heroku local web
forego | starting web.1 on port 5000
web.1  | Node app is running on port 5000

GitHub Flowについて

かなり雑なメモ

ワークフローのイメージ

f:id:mktktmr:20160126142622p:plain








???








何が言いたいかというと、やたらフォークなどはせずにリポジトリは一つ、作業ごとにブランチ切ってそれをmasterブランチにmergeしようって事

概要

  • masterブランチは常にデプロイ出来るように
  • 作業単位でブランチを切る(作業の粒度は?)
  • ブランチ名は作業内容がわかるように命名する
    • submodules-init-task
    • redis2-transition
    • user-content-cache-key
      ... so on
  • commitの粒度は細かく
    • fix typoならそれだけcommit
    • インデントの修正ならそれだけ修正
      上記作業を一緒にcommitしてはダメです
  • フィードバックやアドバイスを欲しい時もPullRequestする
    • PRはmasterへのマージだけが目的ではない
  • マージ後はすぐデプロイ

前提

  • デプロイは自動化
    • 手作業なんて時間の無駄
    • オペレーションミス発生
    • デプロイ中はロックするように
      → 開発スピードが上がると、デプロイする前に次のマージが発生するようになる
      → 不具合が発生した時に、どのデプロイが問題かがわからなくなる
  • テスト重視
    • テストコードがないものはmergeしない
    • テストは自動化

参考資料