### 書寫可重用的跨平臺配置清單
每個系統管理員都夢想著使用統一的、同質的基礎設施,相同的機器都運行相同的操作系統且其版本相同。 正如生活中的其他領域一樣,現實往往是凌亂的、往往與理想情況不符。
你很可能會負責管理一堆使用年限不同、架構不同、運行不同發行版本的不同內核的服務器, 常常是分散在不同的數據中心和不同的互聯網服務供應商(ISP)。
在這種情況下,會造成系統管理員心中對在 for 循環中執行 SSH("SSH in a for loop")的恐懼, 因為在每個服務器上即使執行相同的命令,也可能產生不同的、不可預知的、甚至是危險的結果。
當然,我們應該努力將舊服務器進行更新,并將一切盡可能工作在一個單一的參考平臺之上, 從而使管理更簡單,更廉價,更可靠。但在達到這一目標之前,我們可以使用 Puppet, 它可以使我們更容易地應對異構環境(heterogeneous environments)。
#### 操作步驟
1. 如果你有一些放置在不同數據中心的服務器,這些服務器需要略有不同的網絡配置, 例如,使用節點繼承技術來封裝差異:
```
node wreckspace_server inherits server {
include admin::wreckspace_specific
}
```
2. 你需要應用相同的配置清單到運行著不同 OS 發行版本的服務器,其重要差別可能在于軟件包名、 服務名以及配置文件的存放位置。可以通過在一個類中使用選擇器(selectors) 捕獲這些差異并設置全局變量的值:
```
$ssh_service = $operatingsystem? {
/Ubuntu|Debian/ => "ssh",
default => "sshd",
}
```
之后你就不用擔心配置清單其他部分的差異了,當你要提及這些時, 可以放心的使用變量,它會根據具體環境正確的指向相應的正確值:
```
service { $ssh_service:
ensure => running,
}
```
3. 我們經常需要配合不同的架構;這可能會影響共享庫的路徑,也可能需要不同版本的軟件包。 同樣地,嘗試在一個單一的 architecture 類中封裝所需要的全局變量的值:
```
$libdir = $architecture ? {
x86_64 => "/usr/lib64",
default => "/usr/lib",
}
```
之后在需要一個架構相關的值時,你就可以引用這些全局變量的值, 無論是在配置清單中引用還是在模板中引用均可:
```
; php.ini
[PHP]
; Directory in which the loadable extensions (modules) reside.
extension_dir = <%= libdir %>/php/modules
```
#### 工作原理
這種方法的優點(可以稱為“自上而下”)是你僅需要進行一次選擇。 另一種是自下而上的方法,使用這種方法,在全部配置清單中你隨處可見 使用 selector 或 case 語句的設置:
```
service { $operatingsystem? {
/Ubuntu|Debian/ => "ssh",
default => "sshd" }:
ensure => running,
}
```
這不僅會產生許多重復代碼,而且使代碼難于閱讀。另外,當需要管理一種新的操作系統時, 你必須在所有的配置清單中進行修改,而不是只修改一處。
#### 更多用法
如果你正在為一個公共的發布(例如 Puppet Forge)編寫模塊, 使其盡可能的跨平臺會讓模塊變得更有價值。盡你所能,在不同發布、不同平臺、不同架構上測試, 并且添加適當的變量,使模塊盡可能的應用到各種情況。
如果你正在使用一個公共模塊,并修改它適應自己的環境, 如果你認為你的修改可能會幫助到其他人,可以考慮向公共版本提交你的更新。
即使你不想發布你的模塊,請銘記:一個模塊可能會在生產環境中運行很長一段時間, 并且可能會對其做許多適應環境的改變。 如果從設計模塊的一開始就考慮到這些,那么你(或者最終維護你的代碼人)的生活將會變得更輕松。
> Always code as if the guy who ends up maintaining your code will be a violent psychopath who knows where you live.
>
> — Dave Carhart
#### 參見本書
* 本章的 [使用節點繼承](#ch04sec04) 一節
* 本章的 [使用類的繼承和重載](#ch04sec05) 一節
* 第 9 章的 [使用公共模塊](#ch09sec08) 一節
- 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
- 使用公共模塊
- 使用外部節點分類器
- 創建自定義的資源類型
- 創建自定義的提供者