Chinaunix首页 | 论坛 | 博客
  • 博客访问: 446589
  • 博文数量: 141
  • 博客积分: 211
  • 博客等级: 入伍新兵
  • 技术积分: 1049
  • 用 户 组: 普通用户
  • 注册时间: 2010-09-17 16:25
个人简介

如此经年,望尽千帆。

文章分类

全部博文(141)

文章存档

2014年(73)

2013年(65)

2012年(3)

我的朋友

分类: PERL

2013-02-28 22:17:05

在使用 Gearman 做分布式处理时,各机需要注册一个独立的 job 作为信息反馈,为求方便,Gearman::Worker 脚本 register_function 代码又要通用,于是想到了使用各自的 ip 地址作为 job 命名。

那么怎么在 worker 脚本里获取本机 ip 作为 func 呢?

第一种办法,最简单的,调用 shell:
 $ip = `ifconfig eth0|grep -oE '([0-9]{1,3}\.?){4}'|head -n 1`;
注:这里输入是固定的,所以简单的 [0-9]{1,3} 了,如果是在 web 程序等地方验证 ip,需要更严谨!
或:$ip = `ifconfig eth0|awk -F: '/inet addr/{split($2,a," ");print a[1];exit}'`;
好吧,这样显得太不 perl 了,而且频繁的调用外部 shell 不太好

第二种:

  1. open FH,"ifconfig eth0|";
  2.     while(<FH>){
  3.         last unless /inet addr:((d{1,3}.?){4})/;
  4.         print $1;
  5. }
看起来稍微 perl 了一些,虽然实质跟上面的调用 shell 和 grep 法是一样的。

第三种,更 perl 一点,纯粹读文件:

  1. open FH,'<','/etc/sysconfig/network-scripts/ifcfg-eth0';
  2.     while(<FH>){
  3.         next unless /IPADDRs*=s*(S+)/;
  4.     print $1;
  5. }
进一步的,如果不一定 rh 系,还要去读 /etc/issue ,确定网络配置文件到底是 /etc/sysconfig/network-script/ifcfg-eth0 还是 /etc/network/interfaces 还是其他,然后根据不同发行版写不同的处理方法……额,这是打算自己写模块么?

好吧,大家来充分体会 CPAN 的魅力,去 search 一下,找到一把 Sys::HostIP、Sys::HostAddr、Net::Inetface 等模块。

第四种:


  1. use Sys::HostAddr;
  2.     my $interface = Sys::HostAddr->new(ipv => '4', interface => 'eth0');
  3.     print $interface->main_ip;
不过进去看看pm文件,汗,这几个模块都是调用ifconfig命令,不过是根据发行版的不同进行封装而已。

第五种:
perl -MPOSIX -MSocket -e 'my $host = (uname)[1];print inet_ntoa(scalar gethostbyname($host))';
不过有童鞋说了,这个可能因为hostname的原因,导致获取的都是127.0.0.1……

那么最后还有一招。通过 strace ifconfig 命令可以看到,linux 实质是通过 ioctl 命令完成的网络接口 ip 获取。那么,我们也用 ioctl 就是了!

第六种如下:
 

点击(此处)折叠或打开

  1. #!/usr/bin/perl
  2.     use strict;
  3.     use warnings;
  4.     use Socket;
  5.     require 'sys/ioctl.ph';
  6.     sub get_ip_address($) {
  7.         my $pack = pack("a*", shift);
  8.         my $socket;
  9.         socket($socket, AF_INET, SOCK_DGRAM, 0);
  10.         ioctl($socket, SIOCGIFADDR(), $pack);
  11.         return inet_ntoa(substr($pack,20,4));
  12.     };
  13.     print get_ip_address("eth0");
这样的好处,就是只调用了核心模块,在分发脚本时,不用连带安装其他模块。

注:这个其实是根据网上有的一个 py 的脚本修改的

py版如下:

点击(此处)折叠或打开

  1. #!/usr/bin/python
  2.     import socket
  3.     import fcntl
  4.     import struct
  5.     def get_ip_address(ifname):
  6.         s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
  7.         return socket.inet_ntoa(fcntl.ioctl(
  8.                 s.fileno(),
  9.                 0x8915, # SIOCGIFADDR
  10.                 struct.pack('256s', ifname[:15])
  11.         )[20:24])
  12.     print get_ip_address('eth0')
2012年12月19日增:
为logstash的input/file.rb找到。

ruby版本的:


  1. #!/usr/bin/ruby
  2.     require 'socket'
  3.     SIOCGIFADDR = 0x8915 # get PA address
  4.     def get_ip_address(iface)
  5.       begin
  6.         sock = UDPSocket.new
  7.         buf = [iface,""].pack('a16h16')
  8.         sock.ioctl(SIOCGIFADDR, buf);
  9.         sock.close
  10.         buf[20..24].unpack("CCCC").join(".")
  11.       rescue
  12.         nil
  13.       end
  14.     end
  15.     if $0 == __FILE__
  16.       puts get_ip_address('eth0')
  17.     end
不过看puppet里还是用ifconfig的方法,大家有时间可以搜索下相关内容。


本文原始链接:
阅读(1898) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~