SyntaxHighlighter

About Me

2016年7月14日 星期四

Laravel PhpUnit 關閉 middleware

Laravel Unit test 筆記

Laravel PhpUnit 關閉 middleware

方法一

  • 修改 route 取 middle ware 的方式

於使用 middleware 之前加上

// app/Http/route.php

$middleware = [ ];
if ( !App::runningUnitTests() ) {
    $middleware[] = 'api.auth';
}

$api->version( 'v1', function ( $api )  use ($middleware){
    $api->get( '/users/me', ['middleware' => $middleware,
        'uses' => '\App\Http\Controllers\UserController@currentUser'
     ]);
});

Ref: https://github.com/dingo/api/issues/571#issuecomment-129939376

方法二

  • 於 test case 中加上 use WithoutMiddleware

透過這個方式也會跳過 middleware

<?php

    use Illuminate\Foundation\Testing\WithoutMiddleware;
    use Illuminate\Foundation\Testing\DatabaseMigrations;
    use Illuminate\Foundation\Testing\DatabaseTransactions;

    class ExampleTest extends TestCase
    {
        use WithoutMiddleware;
        
        public function test_example() {
                ...
        }
    }

Ref: https://laravel.com/docs/5.1/testing#application-testing

2016年7月5日 星期二

Mockery 筆記

前言

Mockery 在 UnitTest 上非常常見,需要去模擬第三方套件的時候就會需要用到他,舉例來說像是 PHPMailer ,頭痛的地方有幾點

  1. 寫測試但不希望他寄信
  2. 我該怎麼去模擬他的 method 讓結果跟測試期望的一樣

為了 Coverage 100% 這時候就推薦使用 Mockery 了。

使用 Mockery

安裝套件

composer require --dev mockery/mockery

程式邏輯

  1. overwrite PHPMailer 並 mock 每個使用到的 method
  2. 實作 AddAddress, isHTML, Send 這三個 method
    • AddAddress 參數說明
      • shouldReceive: 需要去 mock 的 method
      • atLeast: 這個 method 至少會執行幾次
      • times(n): 執行 n 次,需搭配 atLeast 去使用
      • withAnyArgs: 沒有限制他的傳入值
    • isHTML 參數說明
      • once: 只會被執行一次
      • with(true): 只會帶入 true
    • Send 參數說明
      • andReturn(true): 只會 return true
  3. (重要)測試單元最後需ㄧㄠ透過 $this->tearDown() 去 reset mockery

案例分享

Mian Class - Main.php

class Main
{
    protected $mail;

    function __construct()
    {
        $this->PHPMailerInit();
    }

    public function run()
    {
        /*
         * Main function .........
         */
        $this->mail->Body = $html;
        if (!$this->mail->Send()) {
            return false;
        }
        return true;
    }

    protected function PHPMailerInit()
    {
        /* phpmailer */
        $this->mail = new \PHPMailer;
        $this->mail->CharSet = 'UTF-8';
        $this->mail->From = 'sender@test.com';
        $this->mail->AddAddress('hashman@test.com');
        $this->mail->AddAddress('andy@test.com');
        $this->mail->isHTML(true);
        $this->mail->Subject = 'My subject';
    }
}


Unit Test - MyTest.php

use \Mockery as m;

class MyTest extends PHPUnit_Framework_TestCase
{
    public function setUp()
    {
        $this->setUpDb();
        $this->mockData();
    }

    public function testMailWillSend()
    {
        $mock = m::mock('overload:\PHPMailer');
        $mock->shouldReceive('AddAddress')
            ->atLeast()
            ->times(1)
            ->withAnyArgs();
        $mock->shouldReceive('isHTML')
            ->once()
            ->with(true);
        $mock->shouldReceive('Send')
            ->andReturn(true);

        $main = new Main();
        $result = $main->run();
        $this->assertEquals(true, $result, 'Email 寄出應該要回傳 true');
        $this->tearDown();
    }

    public function testMailWontSend()
    {
        $mock = m::mock('overload:\PHPMailer');
        $mock->shouldReceive('AddAddress')
            ->atLeast()
            ->times(1)
            ->withAnyArgs();
        $mock->shouldReceive('isHTML')
            ->once()
            ->with(true);
        $mock->shouldReceive('Send')
            ->andReturn(false);

        $main = new Main();
        $result = $main->run();
        $this->assertEquals(false, $result, 'Email 無法寄出應該要回傳 false');
        $this->tearDown();
    }

    protected function tearDown()
    {
        m::close();
    }
}


ps. 依照各自環境的狀況自己去 require autoload.php

參考資料

2016年4月3日 星期日

Laravel Mail 筆記

透過 blade email template 寄 email


$user = User::find(1); // user information

Mail::send('emails.register', compact('user'), function (message) {
    $message->from([from user email account], [from user nickname])
            ->to([receiver email account], [receiver name])
            ->subject('This is email title');
});

不透過 blade email template 寄 email


$user = User::find(1); // user information
$body = 'Something you want to email';

Mail::send([], [], function (message) use($body) {
    $message->from([from user email account], [from user nickname])
            ->to([receiver email account], [receiver name])
            ->subject('This is email title')
            ->setBody($body, 'text/html');
});


如果沒有想要以 html 的格式寄出,可以將 'text/html' 拿掉

參考文件

2015年12月28日 星期一

Git 常用指令

git

整理一下 git 比較常用的一些指令

git

如何使用 git 縮寫

> vim ~/.gitconfig

# 加上下面內容
[alias]
      st = status
      co = checkout
      br = branch
      up = rebase
      ci = commit
      di = diff

git 指令與縮寫後的指令比較

git status   = git st
git checkout = git co
git branch   = git br
git rebase   = git up
git commit   = git ci
git diff     = git di

git 指令(以下都用縮寫)

  1. Status

     git st
    
  2. Branch

     # 顯示目前 local git 的 branch
     git br
    
     # 顯示所有 branch (包含 remote branch))
     git br -a
    
  3. Add

     # add single file
     git add <file_path>
    
     # add all unstage file
     git add .
    
     # 強制新增 ignore file in git
     git add -f [file_path]
    
  4. Commit

     git ci -m "content"
     git ci --amend -c {commit hash}
    
  5. Checkout

     # 切換到任何一個 branch 或是任何一點 commit hash 上
     git co <branch_name>
     git co <commit_hash_number>
    
     # 在現在的 branch 上面新開一個 branch 
     git co -b <new_branch_name>
    
     # checkout 一個 remote branch 
     git co -b <local_branch_name> --track origin/<remote_branch_name>
     EX: git co -b feature/123 --track origin/feature/123
    
     # 清除所有 unstage 檔案到上一次的 commit
     git co -- .
    
  6. Diff

     # 查看目前所有 git 的更動內容
     git diff
    
     # 查看某個檔案的變動
     git diff <file_path>
    
     # 查看 git 更動內容並且忽略空白
     git diff --ignore-all-space
    
  7. Merge

     # 一般直接 merge (ex: feature/123 merge to develop)
     # 建議先 pull 在 merge
     git co develop
     git merge feature/123
    
     # 保留 feature branch 上的線
     # 建議先 pull 在 merge
     git co develop
     git merge --no-ff feature/123
    
  8. Pull

     git pull or git pull origin <branch_name>
    
  9. Push

     # 一般的 push 方法
     git push or git push origin <branch_name>
    
     # 與 tag 一起 push
     git push --follow-tags or git push --follow-tags origin <branch_name>
    
     # Delete remote git branch
     git push origin --delete [branch_name]
    
  10. Fetch

     # 同步目前遠端 git 的狀態
     git fetch origin
    
  11. Tag

     # 顯示目前的 tag 狀態
     git tag
    
     # 在某個 commit hash 上面加上 tag
     git tag -a <tag_name> -m "<tag_content>" <commit_hash>
     EX: git tag -a v1.1.0 -m "test1" a9bf65b
    
     # 查詢 tag 的詳細資料
     git show <tag_name>
     EX: git show v1.1.0
    
  12. Reset

     # reset all include unstage
     git reset -q --hard HEAD --
    
     # hard reset 到某次 commit
     ps. 小心使用,中間的所有 commit 都會消失,必須透過 git reflog 去還原之前的 commit
     git reset --hard <commit_hash>
    
  13. Git Flow

    • 認識 git flow: ihower - Git flow 開發流程
    • 安裝 git flow on mac: git-flow cheatsheet
    • 簡單介紹 git flow 種類
      • feature
        • 從 develop 分支出去,常用於開發新的功能,開發完成後就會 merge 回 develop 上
      • hotfix
        • 從 master 分支出去,常用於 stable 版的緊急修復,修復完成後會 merge 回 master 跟 develop 上
      • release
        • 從 develop 分支出去,用於開發版本要釋出成 stable 版,釋出完畢會 merge 到 master 跟 develop 上

    初始化 git flow

    git flow init

    使用 git flow feature 流程

    git flow feature start 123 // 從 develop 建立一個 feature branch 123 出來 git flow feature publish 123 // 將 feature/123 這個 branch push 到 remote git repo 上 git flow feature finish 123 // 將 feature/123 這個 feature branch 結束後 merge 回 develop 上,並且會將這個 feature branch 移除

    使用 git flow release 流程

    git flow release start v1.1.0 // 從 develop 建立一個 release branch v1.1.0 git flow feature publish v1.1.0 // 將 release/v1.1.0 這個 branch push 到 remote git repo 上 git flow feature finish v1.1.0 // 將 release/v1.1.0 這個 release branch 結束後 merge 進 master 與 develop 上並且壓上版本號,然後將這個 release branch 移除

    使用 git flow hotfix 流程

    git flow hotfix start v1.1.0 // 從 master 建立一個 hotfix branch v1.1.0 git flow hotfix finish v1.1.0 // 將 hotfix/v1.1.0 這個 branch 結束後 merge 回 master 與 develop 上並且壓上版本號,然後將這個 hotfix branch 移除

2015年12月22日 星期二

使用 Mac 安裝 gitbook server

安裝 node.js

安裝 node.js

先到 node js 的官方網站依照自己的 OS 下載 node js 安裝檔
https://nodejs.org/en/

確認 node js 是否安裝

node -v    
npm -v

安裝 gitbook cli

npm install -g gitbook-cli

開啟本機 gitbook 專案

cd [gitbook folder]
# 跳至 gitbook 目錄
gitbook install
# 安裝 gitbook 相關套件
gitbook serve
# 啟動 gitbook server ,預設為 4000 port

2015年12月20日 星期日

CURL 與 Shell Script 相關使用

常用語法

分享一下最近使用 curl 去 access 網站會需要用到的指令

常用語法

1. 使用 curl 直接 access 網站

curl http://xxx.xxx.xxx

2. 透過 curl post data 給網站

curl --data "Account=xxx&Password=xxx" http://xxx.xxx.xxx
// 在這個範例中傳送了 Account 跟 Password 兩個參數

3. 透過 curl 使用 user agent

curl -A "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_11_2) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/47.0.2526.106 Safari/537.36" http://xxx.xxx.xxx
// 此範例為使用 MAC 的 OS 然後還有使用 google chrome 的瀏覽器

4. 透過 curl 去記錄目前所使用的 cookie

1. 取得第一次驗證的 cookie

curl --cookie cookie_file.txt --cookie-jar cookie_file.txt --data "Account=xxx&Password=xxx" http://xxx.xxx.xxx/check.php

// 將驗證資料傳到 check.php 獲得 cookie ,並且將 cookie 存在同層目錄下面的 cookie_file.txt 中

2. 使用已經取得的 cookie file 進行其他的使用

curl --cookie cookie_file.txt http://xxx.xxx.xxx/something.php

shell script 額外使用

1. 使用額外的 config file

  • vim env.conf

      test1="test1"
      test2 = "test2"
      # 宣告會使用到的參數
    
  • vim sourcecode.sh

      source env.conf
    
      # 只要在 source code 中引入這個檔案就可以使用了
    
      echo $test1 # test1
      echo $test2 # test2
    

2. 取得當前執行的目錄

WORKDIR=$(cd "$(dirname "$0")"; pwd)

3. 在 mac 中可以直接雙擊 sh 檔

直接將檔名從 .sh 改成 .command

目前遇到的問題

1. 要傳送兩次 post data 給 check.php 後的 cookie_file 才可以使用

curl --cookie cookie_file.txt --cookie-jar cookie_file.txt --data "Account=xxx&Password=xxx" http://xxx.xxx.xxx/check.php
curl --cookie cookie_file.txt http://xxx.xxx.xxx/something.php

// 這樣會無法在 something.php 中取得到資料

curl --cookie cookie_file.txt --cookie-jar cookie_file.txt --data "Account=xxx&Password=xxx" http://xxx.xxx.xxx/check.php
curl --cookie cookie_file.txt --cookie-jar cookie_file.txt --data "Account=xxx&Password=xxx" http://xxx.xxx.xxx/check.php
curl --cookie cookie_file.txt http://xxx.xxx.xxx/something.php

// 這樣就可以在 something.php 中取得資料

Reference

  1. Linux - shell script 取得當前路徑
  2. 如何雙擊 shell script file
  3. 如何使用 applescript 去呼叫 shell script
  4. 如何使用 curl 進行 session 驗證

2015年12月3日 星期四

Javascript callback function 使用方法

寫法 1

寫 js 也一段時間了
整理一下 js callback funciton 的使用方法

寫法 1

function f1(callback) {
  var data = "123";
  callback && callback(data);
}

function f2() {
  f1(function(result) {
    console.warn("in function");
    console.warn(result);
  });
}

f2();

結果是:

in function
123

寫法 2

function f1(callback) {
  var data = "123";
  if (typeof callback === "function") {
    callback(data);
  }
}

function f2() {
  f1(function(result) {
    console.warn("in function");
    console.warn(result);
  });
}

f2();

結果是:

in function
123

基本上兩者的寫法是一樣的東西
只是寫法 1 稍稍有些簡潔
大家參考一下