### 創建自定義的資源類型
該到你發揮創意的時間了。你已經知道了各種不同的 Puppet 資源類型: 包(package), 文件(file)、用戶(user),等等。 通常情況下,你既可以組合使用這些內置資源類型做你需要做的一切, 又可以通過一個自定義 define 作為一種資源(以內置資源同樣的方式)來使用 (參見第 4 章 [書寫更優質的配置清單](#chapter04) 中有關 define 的內容)。
但是,如果你需要創建自己的資源類型,Puppet 也可以很容易地實現。 原生的資源類型都是使用 Ruby 書寫的,為了創建自己的資源類型,你需要對 Ruby 有一個基本的了解。
讓我們重新回顧一下 **資源類型**(**type**)和 **提供者**(**provider**)之間的區別。 資源類型描述了一個資源和它可擁有的參數(例如,package 類型)。 提供者則告訴 Puppet 如何針對特定的平臺或情況去實現一個資源 (例如,apt/dpkg 提供者為 Debian/Ubuntu 系統實現 package 資源)。
一種類型(如:package)可以有多個提供者(如:apt、yum、fink 等等)。 如果你聲明一個資源時沒有指定提供者,Puppet 會根據環境選擇一個最合適的提供者。
在本節中,我們將看到如何創建一個管理 Git 倉庫的自定義資源類型; 在下一節,我們將編寫一個實現這種資源類型的提供者。
#### 準備工作
1. 在你的 puppet.conf 文件中啟用 pluginsync(若還未啟用):
```
[main]
pluginsync = true
```
2. 在你的 Puppet 倉庫中,為你的插件和類型創建一個自定義模塊(若還不存在):
```
# cd /etc/puppet/modules
# mkdir custom
```
3. 在這個模塊中,創建 lib/puppet/type 目錄:
```
# cd custom
# mkdir -p lib/puppet/type
```
#### 操作步驟
在 type 目錄中創建一個名為 gitrepo.rb 的文件,其內容如下:
```
Puppet::Type.newtype(:gitrepo) do
ensurable
newparam(:source) do
isnamevar
end
newparam(:path)
end
```
#### 工作原理
第一行注冊一個名為 gitrepo 的新類型:
```
Puppet::Type.newtype(:gitrepo) do
```
ensurable 行確保自動給出該類型的屬性(與 Puppet 內置的資源類似):
```
ensurable
```
現在,我們將給出此類型的一些參數。就目前而言,我們所需要的參數分別是: source 參數用于指定 Git 倉庫源的 URL;path 參數用于告訴 Puppet 要在文件系統中的什么位置創建倉庫。
```
newparam(:source) do
isnamevar
end
```
isnamevar 聲明告訴 Puppet 參數 source 是此類型的 namevar。 因此當你聲明這個資源的實例時,你給出的任何名字將被視為 source 的值。例如:
```
gitrepo { "git://github.com/puppetlabs/puppet.git":
path => "/home/john/work/puppet",
}
```
最后,我們添加 path 參數:
```
newparam(:path)
```
#### 更多用法
一旦你熟悉了創建自己的資源類型的方法,你就可以使用自定義的資源類型替換復雜的 exec 資源, 這會使你的配置清單更具可讀性。 然而,通過對自定義資源類型的代碼添加一些文檔和參數校驗使其更強壯更具可重用性是一個好主意。
##### 文檔
我在上面故意舉了一個簡單的例子,但是當你要為生產環境開發真正的自定義類型時, 你應該加入文檔字符串描述類型及其參數的用途。例如:
```
Puppet::Type.newtype(:gitrepo) do
@doc = "Manages Git repos"
ensurable
newparam(:source) do
desc "Git source URL for the repo"
isnamevar
end
newparam(:path) do
desc "Path where the repo should be created"
end
end
```
##### 校驗
當某人試圖向資源傳遞錯誤的值時,你可以使用參數校驗(validate)生成有用的錯誤信息。 例如,你可以校驗要創建倉庫的目錄是否已真實存在:
```
newparam(:path) do
validate do |value|
basepath = File.dirname(value)
unless File.directory?(basepath)
raise ArgumentError , "The path %s doesn't exist" %basepath
end
end
end
```
你也可以為參數指定一個允許的取值列表,例如:
```
newparam(:breakfast) do
newvalues(:bacon, :eggs, :sausages)
end
```
- 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
- 使用公共模塊
- 使用外部節點分類器
- 創建自定義的資源類型
- 創建自定義的提供者