### 使用虛擬資源
什么是虛擬資源(Virtual Resource),我們為什么需要它們? 下面我們來看一個可能會使用虛擬資源的典型例子。 你負責管理 facesquare 和 twitstagram 兩個應用程序,他們都是運行在 Apache 上的 Web 應用程序。 facesquare 的定義看起來可能像這樣:
```
class app::facesquare
{
package { "apache2-mpm-worker": ensure => installed }
}
```
twitstagram 的定義看起來可能像這樣:
```
class app::twitstagram
{
package { "apache2-mpm-worker": ensure => installed }
}
```
一切都很好,直到你需要將兩個應用程序同時應用到單臺服務器上:
```
node micawber
{
include app::facesquare
include app::twitstagram
}
```
現在 Puppet 會出錯,因為你試圖用相同的名字 apache2-mpm-worker 定義兩個 package 資源。 錯誤輸出信息如下:
```
err: Could not retrieve catalog from remote server: Error 400 on SERVER:
Duplicate definition: Package[apache2-mpm-worker] is already defined in
file /etc/puppet/modules/app/manifests/facesquare.pp at line 2; cannot
redefine at /etc/puppet/modules/app/manifests/twitstagram.pp:2 on node
cookbook.bitfieldconsulting.com
```
你可以從其中的一個類中移除重復的包定義,但是這樣話,如果試圖在另一個服務器包含 app 類時,就會因為沒有準備好 Apache 而失敗。
通過在自己的類中放置 Apache 的包資源并使用 include apache 包含它,你就可以解決這個問題, 因為 Puppet 不介意多次包含一個相同的類。但是這有一個缺點,即每個具有潛在沖突的資源都必須有它自己的類。 虛擬資源可以解決這個問題。虛擬資源就像是個普通的資源,特別之處在于它以 @ 字符開始,例如:
```
@package { "apache2-mpm-worker": ensure => installed }
```
你可以把它看作是個 “FYI(僅供參考)” 資源:我只是告訴 Puppet 這個資源存在,但不希望用它做任何事情。 Puppet 將會讀取并記住虛擬資源定義,但實際上不會創建這個資源,直到你明確指出要創建此資源。
要創建這個資源,使用如下的 realize 函數:
```
realize( Package["apache2-mpm-worker"] )
```
對于你想要的資源,可以多次調用 realize 而且不會產生沖突。 因此,虛擬資源用于:當在幾個不同的類中都需要相同的資源,且它們可能會在相同的節點上共存的情況。
#### 操作步驟
1. 創建名為 app 的新模塊:
```
# mkdir -p /etc/puppet/modules/app/manifests
```
2. 使用如下內容創建 /etc/puppet/modules/app/manifests/facesquare.pp 文件:
```
class app::facesquare
{
realize( Package["apache2-mpm-worker"] )
}
```
3. 使用如下內容創建 /etc/puppet/modules/app/manifests/twitstagram.pp 文件:
```
class app::twitstagram
{
realize( Package["apache2-mpm-worker"] )
}
```
4. 使用如下內容創建 /etc/puppet/modules/admin/manifests/virtualpackages.pp 文件:
```
class admin::virtual-packages
{
@package { "apache2-mpm-worker": ensure => installed }
}
```
5. 在一個節點上包含如下代碼:
```
node cookbook
{
include admin::virtual-packages
include app::facesquare
include app::twitstagram
}
```
6. 運行 Puppet。
#### 工作原理
你可以在 admin::virtual-packages 類中定義一個包的虛擬資源。 所有節點都可以包含這個類,并且你可以將所有虛擬資源都放在此類中。 這些虛擬資源都不會實際安裝到節點上,直到你調用 realize:
```
class admin::virtual-packages
{
@package { "apache2-mpm-worker": ensure => installed }
}
```
每個需要 Apache 包的類都可以對虛擬資源調用 realize :
```
class app::twitstagram
{
realize( Package["apache2-mpm-worker"] )
}
```
Puppet 知道如何處理它,因為你設置了相應的虛擬資源,你打算多次引用同一個包, 而不會意外地創建具有相同名子的兩個資源。所以,這正確地實現了我們的需求。
#### 更多用法
為了實現(realize)虛擬資源,你也可以使用 **collection** 語法:
```
Package <| title = "apache2-mpm-worker" |>
```
使用這種語法的好處是,你不僅可以指定資源名,而且可以指定 tag,例如:
```
Package <| tag = "security" |>
```
或者,你可以指定資源類型的所有實例,在查詢部分保留一個空格即可:
```
Package <| |>
```
#### 參見本書
* 本章的 [使用虛擬資源管理用戶](#ch06sec02) 一節
- 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
- 使用公共模塊
- 使用外部節點分類器
- 創建自定義的資源類型
- 創建自定義的提供者