### 從 CSV 文件導入數據
想要知道更多東西嗎?當你需要從表中查找某些值時,可以使用冗長的 case 語句或 selectors 實現,但更整潔的方式是使用 extlookup 函數實現。 在 puppetmaster 上可以使用 extlookup 函數查詢外部的 CSV 文件,并返回匹配的數據片段。
將所有數據組織到一個單一的文件并將它從 Puppet 配置清單中分離出來, 可以使維護工作變得更簡單,也便于與其他人分享:例如, 一個開發團隊通過部署合適的 CSV 文件作為應用程序發布的一部分, 就可以管理 Puppet 需要知道的有關其應用程序的一切。 Puppet 只需要知道在哪里可以找到這個 CSV 文件,剩下的工作交由 extlookup 去完成。
#### 準備工作
1. 使用如下內容創建 /var/www/apps/common.csv 文件:
```
path,/var/www/apps/%{name}
railsversion,3
domain,www.%{name}.com
```
2. 使用如下內容創建 /var/www/apps/myapp.csv 文件:
```
railsversion,2
```
#### 操作步驟
1. 在你的配置清單中添加如下代碼:
```
$extlookup_datadir = "/var/www/apps/"
$extlookup_precedence = [ "%{name}", "common" ]
class app( $name ) {
$railsversion = extlookup("railsversion")
$path = extlookup("path")
$domain = extlookup("domain")
notify { "App data: Path ${path}, Rails version
${railsversion}, domain ${domain}": }
}
class { "app": name => "myapp" }
```
2. 運行 Puppet:
```
# puppet agent --test
info: Retrieving plugin
info: Caching catalog for cookbook.bitfieldconsulting.com
info: Applying configuration version '1303129760'
notice: App data: Path /var/www/apps/myapp, Rails version 2,
domain www.myapp.com
notice: /Stage[main]/App/Notify[App data: Path /var/www/apps/
myapp, Rails version 2, domain www.myapp.com]/message: defined
'message' as 'App data: Path /var/www/apps/myapp, Rails version 2,
domain www.myapp.com'
notice: Finished catalog run in 0.58 seconds
```
#### 工作原理
1. 我們做的第一件事是定義一個變量 $extlookup_datadir,它告訴 extlookup 從哪個目錄查找數據文件。你通常會在 site.pp 或者在你設置全局變量的文件中設置這個變量:
```
$extlookup_datadir = "/var/www/apps/"
```
2. 然后,我們告訴 extlookup 在哪里尋找數據文件,以及尋找的優先順序:
```
$extlookup_precedence = [ "%{name}", "common" ]
```
這可以是任意長度的數組。當我們生成一個 extlookup 查詢時,Puppet 會按順序掃描每個文件,直到找到請求的值。文件名可以包含變量。 我們期望名為 $name 的變量已經被設置,Puppet 將會使用其值作為要查找的第一個文件名。
3. 下一步,在 app 類中,調用 extlookup 獲得一個值:
```
$railsversion = extlookup("railsversion")
```
現在 extlookup 裝置查找一個 CSV 文件并從中讀取數據。這個文件保存在 $extlookup_datadir 目錄(在本例中是 /var/www/apps), 文件名為 %{name}.csv(在本例中是 myapp.csv)。 所以 extlookup 讀取文件的完整路徑是 /var/www/apps/myapp.csv, 此文件包含了 railsversion,2
我們已經找到了所需要的值(2),所以 extlookup 將其返回。
4. 下一個 extlookup 調用就不那么幸運了:
```
$path = extlookup("path")
```
extlookup 還是首先查找文件 myapp.csv,但是它找不到匹配 path 變量的值。 所以會查找 $extlookup_precedence 列表中的下一個文件, 即文件 common.csv:
```
path,/var/www/apps/%{name}
railsversion,3
domain,www.%{name}.com
```
就本例而言返回值為 /var/www/apps/myapp。
你可以看到,這允許我們在 common.csv 文件中設置一個默認值的集合, 每個應用程序都可以選擇其自己的 myapp.csv 文件覆蓋 common.csv 文件中的默認值。 extlookup 會依次查詢列在 $extlookup_precedence 中的文件,直到其找到請求的值。 由于 myapp.csv 被列在首位,所以設置在其中的任何值都優先于文件 common.csv 中的。
#### 更多用法
你也可以在 extlookup 調用中指定默認值,當在所有的 CSV 文件中都沒有找到匹配的數據時就會使用這個默認值:
```
$path = extlookup("path", "/var/www/misc")
```
你還可以指定在查找 $extlookup_precedence 所列的文件之前首先要查找的 CSV 文件:
```
$path = extlookup("path", "/var/www/misc", "paths")
```
這首先在 paths.csv 文件中查找數據,如果沒找到,將會依次查找列在 $extlookup_precedence 中的其他文件。
CSV 文件中的值也可以引用變量,正如我們曾經看到的:
```
domain,www.%{name}.com
```
你可以使用當前范圍內可用的任何變量,包括 Facter 檢測出的 facts 值:
```
domain,%{fqdn}
```
**R.I. Pienaar 的文章 “Complex data and Puppet”** 是對 extlookup 的一個很好的介紹: [http://www.devco.net/archives/2009/08/31/complex_data_and_puppet.php](http://www.devco.net/archives/2009/08/31/complex_data_and_puppet.php) 。
**Jordan Sissel 曾經寫過 “about configuring your whole infrastructure using extlookup”**: [http://sysadvent.blogspot.com/2010/12/day-12-scaling-operabilitywith-truth.html](http://sysadvent.blogspot.com/2010/12/day-12-scaling-operabilitywith-truth.html) 。
#### 參見本書
* 本章的 [導入動態信息](#ch04sec09) 一節
* 第 9 章的 [創建 Facter 的自定義 fact](#ch09sec01) 一節
- 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
- 使用公共模塊
- 使用外部節點分類器
- 創建自定義的資源類型
- 創建自定義的提供者