### 使用 define 資源
> Girl number twenty unable to define a horse!" said Mr. Gradgrind.?—?Charles Dickens.
>
> — Hard Times
除非你知道如何定義你想要什么,不然不會得到預期的結果。 在上一節的示例中,我們看到了如何將同類資源組合成數組從而減少冗余代碼。 然而,這種技術有一個限制,那就是所有的資源必須使用相同的參數。 當你有一組資源擁有一些公用的參數而其中一些資源確有不同的參數時, 就需要使用 define 資源將它們組合在一起。
#### 操作步驟
1. 在配置清單中添加如下代碼:
```
define tmpfile() {
file { "/tmp/$name":
content => "Hello, world",
}
}
tmpfile { ["a", "b", "c"]: }
```
2. 運行 Puppet:
```
notice: /Stage[main]//Node[cookbook]/Tmpfile[a]/File[/tmp/a]/
ensure: defined content as '{md5}bc6e6f16b8a077ef5fbc8d59d0b931b9'
notice: /Stage[main]//Node[cookbook]/Tmpfile[b]/File[/tmp/b]/
ensure: defined content as '{md5}bc6e6f16b8a077ef5fbc8d59d0b931b9'
notice: /Stage[main]//Node[cookbook]/Tmpfile[c]/File[/tmp/c]/
ensure: defined content as '{md5}bc6e6f16b8a077ef5fbc8d59d0b931b9'
```
#### 工作原理
你可以認為 **define** 就像是一個餅干切割器。它描述了一種模式,Puppet 可以用它創建許多類似的資源。任何時候你都可以在你的配置清單中聲明 tmpfile 實例,Puppet 將會插入包含在 tmpfile 定義中的所有資源。
在我們的例子中,名為 tmpfile 的 define 包含了一個 file 資源, 其 content 為 "Hello, world";其 path 為 "/tmp/$name"。 如果你用名字 foo 聲明了一個 tmpfile 實例,如下所示:
```
tmpfile { "foo": }
```
會被 Puppet 要創建的實例名字所代替。這就像是我們創建了一個新的資源類型: tmpfile,它包含一個參數(即其名字)。
與常規資源一樣,我們不僅可以為其傳遞一個字符串的名字,我們還可以傳遞一個數組名, Puppet 會對每一個數組元素創建一個 tmpfile 實例,就像上面的例子那樣。
#### 更多用法
在上面的例子中,我們定義的 define 僅有一個名字參數,不同實例的名字不同。 但是我們可以為其添加任何我們想要的參數,只要我們在 define 中聲明這些參數即可:
```
define tmpfile( $greeting ) {
file { "/tmp/$name":
content => $greeting,
}
}
```
當我們聲明一個資源的實例時,可以為其傳遞參數值,例如:
```
tmpfile{ "foo": greeting => "Hello, world" }
```
你可以使用逗號間隔的列表同時聲明多個參數:
```
define webapp( $domain, $path, $platform ) {
...
}
webapp { "mywizzoapp":
domain => "mywizzoapp.com",
path => "/var/www/apps/mywizzoapp",
platform => "Rails",
}
```
這是對某些常見的資源進行抽象的一項強大的技術,抽象出資源之間的共性并保存在一個地方, 你就**不必每次都做重復勞動**(**Don’t Repeat Yourself**)。 在上面的例子中,webapp 里或許有許多獨立的資源:軟件包(packages)、配置文件(config files)、 源碼檢出(source code checkouts)、虛擬主機(virtual hosts)等等, 但是除了我們為 webapp 提供的參數不同之外,所有實例要執行的工作都相同。 這些可能會在模板中引用,例如:為一個虛擬主機設置域名。
- 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
- 使用公共模塊
- 使用外部節點分類器
- 創建自定義的資源類型
- 創建自定義的提供者