### 有效地分發 cron 任務
當你有許多服務器需要執行相同的 cron 作業時,不在同一時間運行它們通常是個好主意。 如果所有作業都要訪問一個公共服務器,就會給該服務器帶來大量負載, 即使這些服務器不會同時訪問公共服務器,所有服務器也會在同一時間處于繁忙狀態, 這可能會削減它們提供其他服務的能力。
Puppet 的 inline_template 函數允許我們使用 Ruby 的邏輯根據主機名為 cron 作業設置不同的運行時間。
#### 操作步驟
1. 在一個節點中添加如下代碼:
```
define cron_random( $command, $hour )
{
cron { $name:
command => $command,
minute => inline_template("<%= (hostname+name).hash.abs %60 %>"),
hour => $hour,
ensure => "present",
}
}
cron_random { "hello-world":
command => "/bin/echo 'Hello world'",
hour => 2,
}
cron_random { "hello-world-2":
command => "/bin/echo 'Hello world'",
hour => 1,
}
```
2. 運行 Puppet:
```
# puppet agent --test
info: Retrieving plugin
info: Caching catalog for cookbook.bitfieldconsulting.com
info: Applying configuration version '1305713506'
notice: /Stage[main]//Node[cookbook]/Cron_random[hello-world]/
Cron[hello-world]/ensure: created
notice: /Stage[main]//Node[cookbook]/Cron_random[hello-world-2]/
Cron[hello-world-2]/ensure: created
notice: Finished catalog run in 1.07 seconds
```
3. 檢查 crontab 查看是否成功地配置了 cron 作業:
```
# crontab -l
# HEADER: This file was autogenerated at Fri Jul 29 10:58:45 +0000
2011 by puppet.
# HEADER: While it can still be managed manually, it is definitely
not recommended.
# HEADER: Note particularly that the comments starting with
'Puppet Name' should
# HEADER: not be deleted, as doing so could cause duplicate cron
jobs.
# Puppet Name: hello-world
25 2 * * * /bin/echo 'Hello world'
# Puppet Name: hello-world-2
49 1 * * * /bin/echo 'Hello world'
```
#### 工作原理
我們想要為每個 cron 作業選擇一個 **隨機的** 執行分鐘數; 而不是真正的隨機 (或者說,不是每次運行 Puppet 都會改變 cron 作業的運行時間), 但這也或多或少地保證了每個主機上的每個 cron 作業運行時間的不同。
我們可以使用 Ruby 的 hash 方法實現它,它會對任何對象(本例為一個字符串)計算出一個哈希值。 盡管看上去這個哈希值是隨機的,但它每次運行時都相同,所以當再次運行 Puppet 時其值不會改變。
哈希值生成的是一個大整數,而我們想要的是一個 0 到 59 之間的整數,所以我們使用了 Ruby 的 % (模)運算符將其結果限制在這個范圍內。因為只有 60 種可能的值,盡管 hash 函數被設計為盡可能產生隨機的輸出,還是會有些許的碰撞而且這些碰撞對于 minute 應該是均勻分布的。
因為我們希望每個哈希值在不同的主機上是不同的,所以使用主機名做 hash 處理。 然而,我們還希望同一臺主機上的不同作業的哈希值也不同,所以聯合使用了主機名和作業名 (例如 hello-world)做 hash 處理。
#### 更多用法
在本例中,我們僅對 cron 作業的 minute 進行了隨機化,并將 hour 作為 define 定義的一部分。若你同時希望指定要在周幾運行,可以在 cron_random 中添加一個附加參數來指定, 可以像下面這樣為其指定默認值:
```
define cron_random( $command, $hour, $weekday = "*" ) {
```
若你想要對 cron 作業的 hour 進行隨機化(例如:要做的作業可以在一天之內的任何時間執行, 并且必須將它們均勻分布在所有的 24 個小時上),可以對 cron_random 做如下修改:
```
hour => inline_template("<%= (hostname+name).hash.abs % 24 %>"),
```
#### 參見本書
* 第 1 章的 [從 cron 運行 Puppet](#ch01sec05) 一節
- 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
- 使用公共模塊
- 使用外部節點分類器
- 創建自定義的資源類型
- 創建自定義的提供者