2014/05/18

Amazon LinuxのデフォルトのEPELを利用しようとしてエラーが出るときは…

Amazon のよくある質問にも書いてあるのですが、

Q: Extra Packages for Enterprise Linux(EPEL)リポジトリを有効にするにはどうすればよいですか?

これに従って yum --enablerepo=epel install hogehoge とかやると、

One of the configured repositories failed (不明),and yum doesn't have enough cached data to continue. At this point the onlysafe thing yum can do is fail. There are a few ways to work "fix" this:

Cannot retrieve metalink for repository: epel/x86_64. Please verify its path and try again

なんて言われてEPELが利用出来ないときは、/etc/yum.repos.d/epel.repo を
修正してあげるとすんなり直ります。

# vi /etc/yum.repos.d/epel.repo

[epel]
name=Extra Packages for Enterprise Linux 6 - $basearch
baseurl=http://download.fedoraproject.org/pub/epel/6/$basearch
#mirrorlist=https://mirrors.fedoraproject.org/metalink?repo=epel-6&arch=$basearch
failovermethod=priority
enabled=0
gpgcheck=1
gpgkey=file:///etc/pki/rpm-gpg/RPM-GPG-KEY-EPEL-6

↑ もともとコメントアウトされている baseurl のコメントを外して、
 代わりに mirrorlist をコメントアウトする。

2014/05/12

PowerShellを使って、WMI経由でPingをたたく

WMIにはいくつか面白いオブジェクトがあるので使ってみる。

Win32_PingStatus : 指定されたアドレスにPingを飛ばして詳細な結果オブジェクトを返す


試しに www.google.com にPingを投げてみるには、

PS C:\Users\imksoo> $ping_status = Get-WmiObject -Namespace "root\cimv2" -Query "SELECT * FROM Win32_PingStatus WHERE Address='www.google.com'"

PS C:\Users\imksoo> $ping_status | Get-Member


   TypeName: System.Management.ManagementObject#root\cimv2\Win32_PingStatus

Name                           MemberType     Definition
----                           ----------     ----------
PSComputerName                 AliasProperty  PSComputerName = __SERVER
Address                        Property       string Address {get;set;}
BufferSize                     Property       uint32 BufferSize {get;set;}
NoFragmentation                Property       bool NoFragmentation {get;set;}
PrimaryAddressResolutionStatus Property       uint32 PrimaryAddressResolutionStatus {get;set;}
ProtocolAddress                Property       string ProtocolAddress {get;set;}
ProtocolAddressResolved        Property       string ProtocolAddressResolved {get;set;}
RecordRoute                    Property       uint32 RecordRoute {get;set;}
ReplyInconsistency             Property       bool ReplyInconsistency {get;set;}
ReplySize                      Property       uint32 ReplySize {get;set;}
ResolveAddressNames            Property       bool ResolveAddressNames {get;set;}
ResponseTime                   Property       uint32 ResponseTime {get;set;}
ResponseTimeToLive             Property       uint32 ResponseTimeToLive {get;set;}
RouteRecord                    Property       string[] RouteRecord {get;set;}
RouteRecordResolved            Property       string[] RouteRecordResolved {get;set;}
SourceRoute                    Property       string SourceRoute {get;set;}
SourceRouteType                Property       uint32 SourceRouteType {get;set;}
StatusCode                     Property       uint32 StatusCode {get;set;}
Timeout                        Property       uint32 Timeout {get;set;}
TimeStampRecord                Property       uint32[] TimeStampRecord {get;set;}
TimeStampRecordAddress         Property       string[] TimeStampRecordAddress {get;set;}
TimeStampRecordAddressResolved Property       string[] TimeStampRecordAddressResolved {get;set;}
TimestampRoute                 Property       uint32 TimestampRoute {get;set;}
TimeToLive                     Property       uint32 TimeToLive {get;set;}
TypeofService                  Property       uint32 TypeofService {get;set;}
__CLASS                        Property       string __CLASS {get;set;}
__DERIVATION                   Property       string[] __DERIVATION {get;set;}
__DYNASTY                      Property       string __DYNASTY {get;set;}
__GENUS                        Property       int __GENUS {get;set;}
__NAMESPACE                    Property       string __NAMESPACE {get;set;}
__PATH                         Property       string __PATH {get;set;}
__PROPERTY_COUNT               Property       int __PROPERTY_COUNT {get;set;}
__RELPATH                      Property       string __RELPATH {get;set;}
__SERVER                       Property       string __SERVER {get;set;}
__SUPERCLASS                   Property       string __SUPERCLASS {get;set;}
ConvertFromDateTime            ScriptMethod   System.Object ConvertFromDateTime();
ConvertToDateTime              ScriptMethod   System.Object ConvertToDateTime();
IPV4Address                    ScriptProperty System.Object IPV4Address {get=$iphost = [System.Net.Dns]::GetHostEntr...
IPV6Address                    ScriptProperty System.Object IPV6Address {get=$iphost = [System.Net.Dns]::GetHostEntr...

PS C:\Users\imksoo> $status | Format-Table -AutoSize @{Label="Date";Expression={Get-Date -Format "G"}},Address,ResponseTime,ResponseTimeToLive,IPV4Address,IPV6Address

Date                Address        ResponseTime ResponseTimeToLive IPV4Address     IPV6Address
----                -------        ------------ ------------------ -----------     -----------
2014/05/12 11:17:13 www.google.com           81                 52 173.194.117.177

1秒おきにPingを打ち続けるとかであれば、以下の様にFormat-Tableで現在時刻を付与しながら表示してあげると、後々整形しやすいかと。
(ExPingもどきを目指すのであれば、タイムアウト値とかをGet-WmiObjectコマンドレットでPingStatusオブジェクトを作るときに渡しておくとかでしょうか)

PS C:\Users\imksoo> while ( -not (Start-Sleep 1) ) {
>>  $ping_status = Get-WmiObject -Namespace "root\cimv2" -Query "SELECT * FROM Win32_PingStatus WHERE Address='www.googl
e.com'"
>>  $ping_status | Format-Table -AutoSize @{Label="Date";Expression={Get-Date -Format "G"}},Address,ResponseTime,Respons
eTimeToLive,IPV4Address,IPV6Address
>> }
>>

Date                Address        ResponseTime ResponseTimeToLive IPV4Address     IPV6Address
----                -------        ------------ ------------------ -----------     -----------
2014/05/12 11:19:40 www.google.com          112                 52 173.194.117.177

Date                Address        ResponseTime ResponseTimeToLive IPV4Address     IPV6Address
----                -------        ------------ ------------------ -----------     -----------
2014/05/12 11:19:41 www.google.com           94                 52 173.194.117.177

Date                Address        ResponseTime ResponseTimeToLive IPV4Address     IPV6Address
----                -------        ------------ ------------------ -----------     -----------
2014/05/12 11:19:42 www.google.com          179                 52 173.194.117.177

Date                Address        ResponseTime ResponseTimeToLive IPV4Address     IPV6Address
----                -------        ------------ ------------------ -----------     -----------
2014/05/12 11:19:43 www.google.com          203                 52 173.194.117.177

2014/05/07

Windows Server 2008上のPowerShellでHyper-Vの仮想マシンの状態一覧を出すコマンド

WMI+VBScript経由で取得するサンプルスクリプトはいくつか見つけたんですが、PowerShell版はうまく動かなかったりするので書き直したものを残しておきます。

VMStateは仮想マシンの状態を文字列に戻すためのハッシュ変数。
Msvm_ComputerSystemが戻すオブジェクトは仮想マシン+ホストマシンの一覧になっていて、EnabledStateが実際のマシン状態を表す整数値(と見せかけて文字列)を返してくる。
※ EnabledStateをそのままVMState変数に突っ込むと"2"みたいな文字列でハッシュを当たりに行くので、[int]$_.EnabledStateでint型に変換してからVMStateに当たりに行くのが書き換えたポイント。

$VMState=@{
  0="Unknown" ;
  2="Running" ;
  3="Stopped" ;
  32768="Paused" ;
  32769="Suspended";
  32270="Starting" ;
  32771="Snapshotting" ;
  32773="Saving" ;
  32774="Stopping" ;
  32776="Pausing" ;
  32777="Resuming"

Get-WmiObject -Namespace "root\virtualization" -ComputerName "." -Class Msvm_ComputerSystem | Format-Table -autosize @{Label="CurrentDate"; expression={Get-Date -Format "G"}}, @{Label="ElementName"; expression={$_.ElementName}}, @{Label ="EnabledState"; expression={$VMState[[int]$_.EnabledState]}}, Caption, Description

PS C:\Windows\system32> $VMState=@{
  0="Unknown" ;
  2="Running" ;
  3="Stopped" ;
  32768="Paused" ;
  32769="Suspended";
  32270="Starting" ;
  32771="Snapshotting" ;
  32773="Saving" ;
  32774="Stopping" ;
  32776="Pausing" ;
  32777="Resuming"


PS C:\Windows\system32> Get-WmiObject -Namespace "root\virtualization" -ComputerName "." -Class Msvm_ComputerSystem | Format-Table -autosize @{Label="CurrentDate"; expression={Get-Date -Format "G"}}, @{Label="ElementName"; expression={$_.ElementName}}, @{Label ="EnabledState"; expression={$VMState[[int]$_.EnabledState]}}, Caption, Description

CurrentDate         ElementName EnabledState Caption                        Description
-----------         ----------- ------------ -------                        -----------
2014/05/07 11:53:11 TESTHV01    Running      ホスト コンピューター システム Microsoft ホスト コンピューター システム
2014/05/07 11:53:11 HOGEHOGE    Stopped      仮想マシン                     Microsoft Virtual Machine
2014/05/07 11:53:11 TESTPC1     Stopped      仮想マシン                     Microsoft Virtual Machine

2014/05/07 11:53:11 TESTPC2     Running      仮想マシン                     Microsoft Virtual Machine


Windows Server 2012 以降であれば、標準搭載のHyper-VコマンドレットでGet-VM一発で終わるんですけどねぇ…。

PS C:\Users\imksoo> Get-VM

Name               State CPUUsage(%) MemoryAssigned(M) Uptime   Status
----               ----- ----------- ----------------- ------   ------
CentOS6.2-minimal  Off   0           0                 00:00:00 正常稼働中
CentOS6.2-minimal2 Off   0           0                 00:00:00 正常稼働中
CentOS6.3          Off   0           0                 00:00:00 正常稼働中
Chef-Server        Off   0           0                 00:00:00 正常稼働中

2014/04/07

Pacemaker + cmanでクラスタを作る手順メモ

日曜日の夜が更けてきたのでコマンドだけメモして寝るw

# vi /etc/hosts
10.3.12.121     LAINCL01.local LAINCL01
10.3.12.122     LAINCL02.local LAINCL02


# yum install pacemaker corosync pcs cman ccs
# corosync-keygen
# cd /etc/corosync/
# scp authkey root@10.3.12.122:/etc/corosync
# cp corosync.conf.example.udpu corosync.conf
# vi corosync.conf
以下の設定を記載
・interface memberにノードIPを並べる
・bindnetaddrはネットワークアドレス
・transportはudpuにする
・logging to_syslogはnoにする


# iptables -I INPUT 1 --protocol udp --dport 5405 -j ACCEPT
# iptables -I INPUT 1 --protocol udp --sport 5404 -j ACCEPT
# iptables -I OUTPUT 1 --protocol udp --dport 5405 -j ACCEPT
# iptables -I OUTPUT 1 --protocol udp --sport 5404 -j ACCEPT


# crm_mon

# service cman start
# service pacemaker start
# chkconfig cman on
# chkconfig pacemaker on

# ccs -f /etc/cluster/cluster.conf --createcluster LAINCL00
# ccs -f /etc/cluster/cluster.conf --addnode LAINCL01.local
# ccs -f /etc/cluster/cluster.conf --addnode LAINCL02.local
# ccs -f /etc/cluster/cluster.conf --addfencedev pcmk agent=fence_pcmk
# ccs -f /etc/cluster/cluster.conf --addmethod pcmk-redirect LAINCL01.local
# ccs -f /etc/cluster/cluster.conf --addmethod pcmk-redirect LAINCL02.local
# ccs -f /etc/cluster/cluster.conf --addfenceinst pcmk LAINCL01.local pcmk-redirect port=LAINCL01.local
# ccs -f /etc/cluster/cluster.conf --addfenceinst pcmk LAINCL02.local pcmk-redirect port=LAINCL02.local

# echo "CMAN_QUORUM_TIMEOUT=0" >> /etc/sysconfig/cman

# pcs property set stonith-enabled=false
# pcs property set no-quorum-policy=ignore

参考:
http://clusterlabs.org/quickstart-redhat.html
http://blog.suz-lab.com/2012/12/corosync-pacemaker.html
http://linux-ha.sourceforge.jp/wp/dl/pminstall_cent5
http://sios-oss.blogspot.jp/2013/05/pacemaker-corosync-ha.html

2014/02/07

Windows + PowerShellで特定の引数を持ったプロセスを見つけて、メモリ使用量を取るコマンド

書くほどではないがメモとして。

PowerShellではGet-Processコマンドレットでプロセス一覧が取得出来る。
ただ、それだとコマンドライン引数が分からないため、「hoge.jarを動かしているjava.exe」みたいなプロセスを特定することが出来ない。

そのためGet-WmiObject経由で Win32_Process オブジェクトを使ってコマンドライン引数を含めた情報を取得してみた。

Get-WmiObjectを利用した場合

PS C:\> Get-WmiObject Win32_Process | ? { $_.Name -eq 'notepad.exe' -and $_.CommandLine -ilike '*test.txt*' }


__GENUS                    : 2
__CLASS                    : Win32_Process
__SUPERCLASS               : CIM_Process
__DYNASTY                  : CIM_ManagedSystemElement
__RELPATH                  : Win32_Process.Handle="5124"
__PROPERTY_COUNT           : 45
__DERIVATION               : {CIM_Process, CIM_LogicalElement, CIM_ManagedSystemElement}
__SERVER                   : SL20807-W8
__NAMESPACE                : root\cimv2
__PATH                     : \\SL20807-W8\root\cimv2:Win32_Process.Handle="5124"
Caption                    : notepad.exe
CommandLine                : "C:\WINDOWS\system32\notepad.exe" c:\test.txt
CreationClassName          : Win32_Process
CreationDate               : 20140207164215.299105+540
CSCreationClassName        : Win32_ComputerSystem
CSName                     : SL20807-W8
Description                : notepad.exe
ExecutablePath             : C:\WINDOWS\system32\notepad.exe
ExecutionState             :
Handle                     : 5124
HandleCount                : 122
InstallDate                :
KernelModeTime             : 1093750
MaximumWorkingSetSize      : 1380
MinimumWorkingSetSize      : 200
Name                       : notepad.exe
OSCreationClassName        : Win32_OperatingSystem
OSName                     : Microsoft Windows 8.1 Pro|C:\WINDOWS|\Device\Harddisk0\Partition2
OtherOperationCount        : 343
OtherTransferCount         : 866
PageFaults                 : 8697
PageFileUsage              : 4368
ParentProcessId            : 7260
PeakPageFileUsage          : 4572
PeakVirtualSize            : 173985792
PeakWorkingSetSize         : 13676
Priority                   : 8
PrivatePageCount           : 4472832
ProcessId                  : 5124
QuotaNonPagedPoolUsage     : 12
QuotaPagedPoolUsage        : 311
QuotaPeakNonPagedPoolUsage : 14
QuotaPeakPagedPoolUsage    : 330
ReadOperationCount         : 31
ReadTransferCount          : 23364
SessionId                  : 6
Status                     :
TerminationDate            :
ThreadCount                : 1
UserModeTime               : 312500
VirtualSize                : 167088128
WindowsVersion             : 6.3.9600
WorkingSetSize             : 13975552
WriteOperationCount        : 0
WriteTransferCount         : 0
PSComputerName             : SL20807-W8
ProcessName                : notepad.exe
Handles                    : 122
VM                         : 167088128
WS                         : 13975552
Path                       : C:\WINDOWS\system32\notepad.exe

別解:WMICコマンドを使った場合

C:\> wmic.exe process where "(Name='notepad.exe') and (CommandLine like '%test.txt%') list /format:list

CommandLine="C:\WINDOWS\system32\notepad.exe" c:\test.txt
CSName=SL20807-W8
Description=notepad.exe
ExecutablePath=C:\WINDOWS\system32\notepad.exe
ExecutionState=
Handle=5124
HandleCount=122
InstallDate=
KernelModeTime=1093750
MaximumWorkingSetSize=1380
MinimumWorkingSetSize=200
Name=notepad.exe
OSName=Microsoft Windows 8.1 Pro|C:\WINDOWS|\Device\Harddisk0\Partition2
OtherOperationCount=343
OtherTransferCount=866
PageFaults=8697
PageFileUsage=4368
ParentProcessId=7260
PeakPageFileUsage=4572
PeakVirtualSize=173985792
PeakWorkingSetSize=13676
Priority=8
PrivatePageCount=4472832
ProcessId=5124
QuotaNonPagedPoolUsage=12
QuotaPagedPoolUsage=311
QuotaPeakNonPagedPoolUsage=14
QuotaPeakPagedPoolUsage=330
ReadOperationCount=31
ReadTransferCount=23364
SessionId=6
Status=
TerminationDate=
ThreadCount=1
UserModeTime=312500
VirtualSize=167088128
WindowsVersion=6.3.9600
WorkingSetSize=13975552
WriteOperationCount=0
WriteTransferCount=0

遠回りだけど何とかパフォーマンスカウンタでやりたい場合

ちなみに、typeperfコマンドやパフォーマンスカウンタを使いたい場合、対象となるインスタンスを特定するために一工夫が必要になる。

エレガントにやる手順は Find a Performance Counter Instance by a Process ID に詳しいのでそちらを参照してみてください。
キモはPowerShellだと ImportCSV コマンドレットでCSVデータを名前付きオブジェクトで扱えるから、それでプロセスIDを元にインスタンス名を特定できるところ。

2014/01/31

AWS CloudWatchの課金情報をZabbixのLow Level Discoveryで見られるようにする #1

AWSのCloudWatchでは、EC2インスタンスの性能情報と同様に、AWSアカウントに紐付く課金情報(AWS/Billing)をAPI経由で取得することが出来ます。
今回はまずCloudWatchで課金情報項目を引っ張り出して、ZabbixのLow Level Discoveryで拾えるように加工するところまでやってみます。

AWS CLIをインストール

今回は、オンプレミス環境に置いてあるCentOS 6上から監視するイメージです。
まずはAWS APIを利用するために、AWSコマンドラインインターフェース ツール(AWS CLI)を導入します。
AWS CLIはPython pip経由でインストールするのが手っ取り早いので、EPELのpython-pip経由で導入。

 # yum install python-pip
 # python-pip awscli
 # aws configure

jqのインストール

AWS CLIの出力は基本的にJSONフォーマットになるので、整形したり編集したりをコマンドラインで出来る jq コマンドを導入します。

 # yum install jq

ZabbixのLow Level Discoveryが受け付けるJSONフォーマット

Zabbix 2.0 Manual - 3.4. Creating custom LLD rules を見ると分かる…と思うのですが、簡単にまとめてみました。
  • 最上位となる要素は辞書(Dictionary)。
    • 内容は キー "data"、値はディスカバリした結果を格納した配列(Array)。
    • 例)
      { "data": [ディスカバリー結果] }
  • [ディスカバリ結果]は、それぞれZabbixマクロの変数名と同じキーと、それに対する値を持つ辞書の配列。
    • 例)
       [{ "{#FS_NAME}":"/", "{#FS_TYPE}":"ufs" },  { "{#FS_NAME}":"/boot", "{#FS_TYPE}":"ufs" }]
    • 変数名に使えるのは "A-Z , 0-9 , _ , ."のいずれか。
      ※英小文字は使えないので注意。
  • 組み立ててみて、下記のような感じになるとOK
    { "data": [  { "{#FS_NAME}":"/", "{#FS_TYPE}":"ufs" },  { "{#FS_NAME}":"/boot", "{#FS_TYPE}":"ufs" }
    ]
     }

AWS CloudWatchの課金情報メトリックのフォーマット

AWS CLI経由で課金情報のメトリック一覧を出力させてみます。
※その月に利用したサービスに応じて、出力されるメトリック(項目)は増減します。

# aws cloudwatch --region us-east-1 list-metrics | jq "." | head -n 30
{
  "Metrics": [
    {
      "MetricName": "EstimatedCharges",
      "Dimensions": [
        {
          "Value": "AmazonEC2",
          "Name": "ServiceName"
        },
        {
          "Value": "USD",
          "Name": "Currency"
        }
      ],
      "Namespace": "AWS/Billing"
    },
    {
      "MetricName": "EstimatedCharges",
      "Dimensions": [
        {
          "Value": "AmazonRoute53",
          "Name": "ServiceName"
        },
        {
          "Value": "USD",
          "Name": "Currency"
        }
      ],
      "Namespace": "AWS/Billing"
    },
    …

ざっと見た感じ、以下のように変換すると良さそうです。

AWS CLIJSON Zabbix LLDJSON
.Metrics .Data
.Namespace {#AWS_BILLING_NAMESPACE}
.MetricName {#AWS_BILLING_METRIC_NAME}
(.MetricName + .DimensionsServiceName) {#AWS_BILLING_METRIC_DISPLAYNAME}
(.Dimensions | tostring) {#AWS_BILLING_METRIC_DIMENSIONS}

というわけで、jqコマンドで変換するコマンドを組み立ててみる。。。
(試行錯誤すること 2時間ほど…)

できました!

 # aws cloudwatch --region us-east-1 list-metrics | jq '.Metrics[] | {
 "{#AWS_BILLING_NAMESPACE}": (.Namespace),
 "{#AWS_BILLING_METRIC_NAME}": (.MetricName),
 "{#AWS_BILLING_METRIC_DISPLAYNAME}": (.MetricName + " - " + ((.Dimensions[] | select(.Name=="ServiceName") | .Value) // "total") ),
 "{#AWS_BILLING_METRIC_DIMENSIONS}": (.Dimensions | tostring)
}' | jq -s '{data:(.|sort)}'

ためしてみる!

# aws cloudwatch --region us-east-1 list-metrics | jq '.Metrics[] | {
 "{#AWS_BILLING_NAMESPACE}": (.Namespace),
 "{#AWS_BILLING_METRIC_NAME}": (.MetricName),
 "{#AWS_BILLING_METRIC_DISPLAYNAME}": (.MetricName + " - " + ((.Dimensions[] | select(.Name=="ServiceName") | .Value) // "total") ),
 "{#AWS_BILLING_METRIC_DIMENSIONS}": (.Dimensions | tostring)
 }' | jq -s '{data:(.|sort)}'

{
  "data": [
    {
      "{#AWS_BILLING_NAMESPACE}": "AWS/Billing",
      "{#AWS_BILLING_METRIC_NAME}": "EstimatedCharges",
      "{#AWS_BILLING_METRIC_DISPLAYNAME}": "EstimatedCharges - AWSDataTransfer",
      "{#AWS_BILLING_METRIC_DIMENSIONS}": "[{\"Value\":\"AWSDataTransfer\",\"Name\":\"ServiceName\"},{\"Value\":\"USD\",\"Name\":\"Currency\"}]"
    },
    {
      "{#AWS_BILLING_NAMESPACE}": "AWS/Billing",
      "{#AWS_BILLING_METRIC_NAME}": "EstimatedCharges",
      "{#AWS_BILLING_METRIC_DISPLAYNAME}": "EstimatedCharges - AmazonEC2",
      "{#AWS_BILLING_METRIC_DIMENSIONS}": "[{\"Value\":\"AmazonEC2\",\"Name\":\"ServiceName\"},{\"Value\":\"USD\",\"Name\":\"Currency\"}]"
    },
    {
      "{#AWS_BILLING_NAMESPACE}": "AWS/Billing",
      "{#AWS_BILLING_METRIC_NAME}": "EstimatedCharges",
      "{#AWS_BILLING_METRIC_DISPLAYNAME}": "EstimatedCharges - AmazonRoute53",
      "{#AWS_BILLING_METRIC_DIMENSIONS}": "[{\"Value\":\"AmazonRoute53\",\"Name\":\"ServiceName\"},{\"Value\":\"USD\",\"Name\":\"Currency\"}]"
    }, …

お、上手くいっているっぽい。

ちなみに上記の情報を基に、実際の課金情報を引っ張ってみるとこうなります。

 # aws cloudwatch --region us-east-1 get-metric-statistics \
    --namespace "AWS/Billing" \
    --metric-name "EstimatedCharges" \
    --dimensions "[{\"Value\":\"AmazonEC2\",\"Name\":\"ServiceName\"},{\"Value\":\"USD\",\"Name\":\"Currency\"}]" \
    --period 60 \
    --start-time $(date --iso-8601=seconds --date '24 hour ago') \
    --end-time   $(date --iso-8601=seconds) \
    --statistics "Maximum" \
     | jq '.Datapoints | sort_by(.Timestamp)'

[
  {
    "Unit": "None",
    "Maximum": 1.19,
    "Timestamp": "2014-01-30T11:34:00Z"
  },
  {
    "Unit": "None",
    "Maximum": 1.2,
    "Timestamp": "2014-01-30T15:34:00Z"
  },
  {
    "Unit": "None",
    "Maximum": 1.2,
    "Timestamp": "2014-01-30T19:34:00Z"
  },
  {
    "Unit": "None",
    "Maximum": 1.21,
    "Timestamp": "2014-01-30T23:34:00Z"
  },
  {
    "Unit": "None",
    "Maximum": 1.21,
    "Timestamp": "2014-01-31T03:34:00Z"
  },
  {
    "Unit": "None",
    "Maximum": 1.21,
    "Timestamp": "2014-01-31T07:34:00Z"
  }
]

実際には最後の値しか要らなさそうなので、jqコマンドに "reverse | .[0]"を追加することになりそうです。( .[-1]だと上手くいかない )
続きはまた今度。

AWSのELBがなんで動的IPなのか考えてみた。

最近、AWSを触る日々。

ふつーの環境でBIG-IPだか、NetScalerだかってロードバランサーを使ってると不思議なのは、
「ELB (Elastic Load Balancing)はエンドポイント名=FQDNは固定なのに、なんでIPアドレスは動的なの?」
ってところ。

# ちなみにさらっと調べてみた感じ、EIPをELBに割り当てることも出来ないみたいですね。。。


で、AWSの世界でELBを作るとしたらこうなのかな?とすると納得というモデルを考えてみた。


こういうもんだととらえれば、あえてなぜ代表IPを固定にしてないのか分かりますね。

古典的なロードバランサが代表IPを固定に出来るのは、前提としてネットワークがマルチキャストを通してたり、STPやStackされたL2スイッチに横並びに配置されることを想定しているからで、
AWSでマルチAZ構成だったりすればそういう前提は成り立たないですから、その中で出来る構成と言えばDNSラウンドロビン+リバースプロキシのような構成だと思います。

ま、ここまで勝手な妄想なので本当かどうかは知りませんが。

↓この辺を読んでも、なぜなのかははっきり書いていません(当然ですね・・・