### 當文件更新時運行命令
當某個特定的文件更新后 Puppet 就該采取一些行動,這是一個非常常見的模式。 例如,在 rsync 配置片段的例子中,一旦修改了某個片段文件,就會調用 exec 資源更新主配置文件 rsyncd.conf。
每次運行 Puppet,exec 資源都會被運行,除非指定了如下參數中的一個:
* creates
* onlyif
* unless
* refreshonly => true
refreshonly 參數的意思是:僅當從其他資源(例如一個文件資源)獲得一個 notify 才執行 exec 資源。
#### 準備工作
安裝 nginx 包(實際上,我們只需要配置文件,但這是獲得它的最簡單方式):
```
# apt-get install nginx
```
#### 操作步驟
1. 創建一個 nginx 模塊,其目錄結構如下:
```
# mkdir /etc/puppet/modules/nginx
# mkdir /etc/puppet/modules/nginx/files
# mkdir /etc/puppet/modules/nginx/manifests
```
2. 使用如下內容創建 /etc/puppet/modules/nginx/manifests/nginx.pp 文件:
```
class nginx {
package { "nginx": ensure => installed }
service { "nginx":
enable => true,
ensure => running,
}
exec { "reload nginx":
command => "/usr/sbin/service nginx reload",
require => Package["nginx"],
refreshonly => true,
}
file { "/etc/nginx/nginx.conf":
source => "puppet:///modules/nginx/nginx.conf",
notify => Exec["reload nginx"],
require => Package["nginx"],
}
}
```
3. 復制系統中的 nginx.conf 文件到模塊的相應目錄:
```
cp /etc/nginx/nginx.conf /etc/puppet/modules/nginx/files
```
4. 添加如下代碼到你的配置清單:
```
include nginx
```
5. 對復制到 Puppet 中的 nginx.conf 文件做個小改動,以便進行后續的測試:
```
# echo \# >>/etc/puppet/modules/nginx/files/nginx.conf
```
6. 運行 Puppet:
```
# puppet agent --test
info: Retrieving plugin
info: Caching catalog for cookbook.bitfieldconsulting.com
info: Applying configuration version '1303745502'
--- /etc/nginx/nginx.conf 2010-02-15 00:16:47.000000000 -0700
+++ /tmp/puppet-file20110425-31239-158xcst-0 2011-04-25
09:39:49.586322042 -0600
@@ -48,3 +48,4 @@
# proxy on;
# }
# }
+#
info: FileBucket adding /etc/nginx/nginx.conf as {md5}7bf139588b5e
cd5956f986c9c1442d44
info: /Stage[main]/Nginx/File[/etc/nginx/nginx.conf]:
Filebucketed /etc/nginx/nginx.conf to puppet with sum
7bf139588b5ecd5956f986c9c1442d44
notice: /Stage[main]/Nginx/File[/etc/nginx/nginx.conf]/content:
content changed '{md5}7bf139588b5ecd5956f986c9c1442d44' to '{md5}
d28d08925174c3f6917a78797c4cd3cc'
info: /Stage[main]/Nginx/File[/etc/nginx/nginx.conf]: Scheduling
refresh of Exec[reload nginx]
notice: /Stage[main]/Nginx/Exec[reload nginx]: Triggered 'refresh'
from 1 events
notice: Finished catalog run in 1.69 seconds
```
#### 工作原理
對于大多數服務來說,你應該簡單地定義一個服務資源,它從配置文件獲得 notify。 這會使 Puppet 重啟服務,從而應用改變之后的更新。
然而,nginx 有時不能正確的重新啟動,尤其是當使用 Puppet 重新啟動時, 所以我為某個站點炮制了一個讓 Puppet 運行 /etc/init.d/nginx reload 的補救措施, 用它替代服務的重啟。下面闡述它是如何工作的。
將 exec 資源中的 refreshonly 參數設置成 true:
```
exec { "reload nginx":
command => "/usr/sbin/service nginx reload",
require => Package["nginx"],
refreshonly => true,
}
```
所以,僅當它獲得通知才會運行。
如果配置文件發生改變,就提供所需的通知(notify):
```
file { "/etc/nginx/nginx.conf":
source => "puppet:///modules/nginx/nginx.conf",
notify => Exec["reload nginx"],
}
```
每當 Puppet 更新了這個配置文件,它就會運行 exec, 這將會調用如下命令重新加載修改后的配置文件:
```
/usr/sbin/service nginx reload
```
如果服務支持 reload 命令參數,就會在不中斷服務的情況下為 daemon 發出一個重新讀取配置文件的進程信號。
實際上,對于本例,更好的方法是為 nginx 服務定義如下的 restart 命令:
```
service { "nginx":
restart => "/etc/init.d/nginx reload",
}
```
但是我想要和你分享一些我寫的用于展示 notify -> Exec 技術的實際代碼, 并且那時,我還不知道 restart 而且它也不存在。 不過作為一種通用的模式,當更新文件后需要采取某種行動時,你會發現它有用。
#### 更多用法
每當遇到資源更新就要采取某些行動的情況,你就可以使用這個類似的模式。可能的用途包括:
* 觸發服務重新加載配置文件
* 運行語法檢查,然后再重新啟動服務
* 連接 config 片段
* 運行測試
* 鏈接 exec 資源
如果當一個文件(假設名為 somefile)更新后需要執行多個命令,那么在每個命令的 exec 資源聲明中都使用 subscribe => File[somfile], (即一旦檢測到 somefile 文件的變化就重新執行 exec 資源)會更簡單, 而不是在 file 資源中使用 notify 執行命令。 效果是一樣的。
>  譯者注
> 你也可以在 service 資源定義中使用 subscribe, 例如在一個 Redhat 風格的系統中,你可以使用如下的 service 聲明:
> ```
> service { "nginx":
> enable => true,
> ensure => running,
> restart => "/etc/init.d/nginx reload",
> subscribe => [ File["/etc/nginx/nginx.conf"],
> File["/etc/sysconfig/nginx"] ]
> }
>
> ```
> 在上例中,使用數組為 subscribe 指定了兩個要檢測的文件, 也就是說,一旦檢測到兩個文件中的任何一個發生變化就重啟服務。
- Puppet 2.7 Cookbook 中文版
- 中文翻譯版
- 譯者序
- 項目緣起
- 翻譯方法
- 社區鏈接
- 社區建議
- 貢獻者
- 原書版權頁
- 關于作者
- 前言
- 本書內容
- 閱讀前提
- 適用讀者
- 格式約定
- 讀者反饋
- 客戶支持
- 下載案例代碼
- 勘誤表
- Puppet 基礎設施
- 使用版本控制
- 使用提交鉤子
- 使用 Rake 部署變更
- 配置 Puppet 的文件服務器
- 從 cron 運行 Puppet
- 使用自動簽名
- 預簽名證書
- 從 Puppet 的 filebucket 檢索文件
- 使用 Passenger 擴展 Puppet 的部署規模
- 創建去中心化的分布式 Puppet 架構
- 監控、報告和排錯
- 生成報告
- 通過 Email 發送包含特定標簽的日志信息
- 創建圖形化報告
- 自動生成 HTML 文檔
- 繪制依賴關系圖
- 測試你的 Puppet 配置清單
- 執行模擬運行
- 檢測編譯錯誤
- 理解 Puppet 的錯誤信息
- 顯示命令的輸出結果
- 輸出調試信息
- 檢查配置設置
- 使用標簽
- 使用運行階段
- 使用不同的環境
- Puppet 語言及其寫作風格
- 使用 Puppet 社區規范
- 使用模塊
- 使用標準的命名規范
- 使用嵌入式 Ruby 代碼
- 使用純 Ruby 代碼書寫配置清單
- 遍歷多個項目
- 書寫強大的條件語句
- 在 if 語句中使用正則表達式
- 使用選擇器和 case 語句
- 檢測字符串中是否包含指定的值
- 使用正則表達式替換
- 書寫更優質的配置清單
- 使用資源的數組
- 使用 define 資源
- 指定資源的依賴關系
- 使用節點繼承
- 使用類的繼承和重載
- 給類傳遞參數
- 書寫可重用的跨平臺配置清單
- 獲得系統的環境信息
- 導入動態信息
- 從 CSV 文件導入數據
- 給 Shell 命令傳遞參數
- 使用文件和軟件包
- 為配置文件添加配置行
- 使用 Augeas 自動修改配置文件
- 使用配置片段構建配置文件
- 使用 ERB 模板
- 在模板中遍歷數組
- 從第三方倉庫安裝軟件包
- 配置 APT 軟件倉庫
- 配置 GEM 倉庫
- 從源碼包自動構建軟件
- 比較軟件包的版本
- 用戶和虛擬資源
- 使用虛擬資源
- 使用虛擬資源管理用戶
- 管理用戶基于密鑰的 SSH 訪問
- 管理用戶的自定義文件
- 有效地分發 cron 任務
- 當文件更新時運行命令
- 使用主機資源
- 為文件資源指定多個源
- 使用文件資源遞歸地分發整個目錄樹
- 清理過期的舊文件
- 使用日程表資源
- 資源的審計
- 臨時禁用資源
- 管理時區
- 應用程序
- 管理 Apache 服務
- 創建 Apache 虛擬主機
- 創建 Nginx 虛擬主機
- 創建 MySQL 數據庫及用戶
- 管理 Drupal 站點
- 管理 Rails 應用程序
- 服務器和云基礎設施
- 部署 Nagios 監控服務器
- 使用 Heartbeat 構建高可用服務
- 管理 NFS 服務和文件共享
- 使用 HAProxy 為多個 web 服務器實現負載均衡
- 使用 iptables 管理防火墻
- 管理 Amazon 的 EC2 實例
- 使用 Vagrant 管理虛擬機
- 外部工具和 Puppet 生態環境
- 創建 Facter 的自定義 fact
- 在運行 Puppet 之前和之后執行命令
- 從 Shell 會話生成 Puppet 配置清單
- 從運行的系統上生成 Puppet 配置清單
- 使用 Puppet Dashboard
- 使用 Foreman
- 使用 MCollective
- 使用公共模塊
- 使用外部節點分類器
- 創建自定義的資源類型
- 創建自定義的提供者