海运的博客

openwrt使用dnspod api自动更新ddns

发布时间:February 12, 2019 // 分类:OpenWrt // No Comments

通过dnspod api实现动态ddns更新ip,ipv4和ipv6支持,shell脚本如下:

#!/bin/bash
token="www.haiyun.me"
domain="haiyun.me"
if which jq > /dev/null; then
  json="jq"
elif which jsonfilter > /dev/null; then
  json="jsonfilter"
else
  echo 'please install jq or jsonfilter'
  exit
fi
if ! which curl > /dev/null || ! which curl > /dev/null; then
  echo 'please install curl and grep'
  exit
fi
if [[ $1 == "list" ]]; then
  curl -s -d "login_token=$token&format=json&domain=$domain" "https://dnsapi.cn/Record.List" | jq -r -M '.records[]|.name + "\t\t " + .type + "\t\t " + .value'
  exit
fi
if [[ $1 == "delete" ]]; then
  if [[ ! $3 || ! $2 ]]; then
    echo 'use ddns.sh delete name type'
    exit
  fi
  id=$(curl -s -d "login_token=$token&format=json&domain=$domain" "https://dnsapi.cn/Record.List" | jq -r -e ".records | .[] | select(.name == \"$2\" and .type == \"${3^^}\")|.id")
  if [[ $id ]]; then
    if curl -s -d "login_token=$token&format=json&domain=$domain&record_id=$id" https://dnsapi.cn/Record.Remove | grep -q '"code":"1"'; then
      echo "sus"
    fi
  else
    echo 'no record'
  fi
  exit
fi
if [[ ! $1 || ! $2 ]]; then
  echo 'use ddns.sh name ip'
  echo 'use ddns.sh list'
  echo 'use ddns.sh delete name type'
  exit
fi
name=$1
new_ip=$2
if [[ $new_ip =~ ^[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}$ ]]; then
  #sleep 10
  #curl http://192.168.168.6/announce.php --silent --output /dev/null
  record_type='A'
  echo 'ipv4'
elif [[ $new_ip =~ ^([0-9a-fA-F]{0,4}:){1,7}[0-9a-fA-F]{0,4}$ ]]; then
  echo 'ipv6'
  record_type='AAAA'
else
  echo "invalid IP address $new_ip"
  #logger -t ddns "invalid IP address $new_ip"
  exit
fi
curl -s -d "login_token=$token&format=json&domain=$domain" "https://dnsapi.cn/Record.List" -o /tmp/dns.txt
if ! grep -q '"code":"1"' /tmp/dns.txt; then
  echo 'get record list error'
  exit
fi

if [[ $record_type == "AAAA" ]]; then
  if [[ $json == "jq" ]]; then
    id=$(jq -r -e ".records | .[] | select(.name == \"$name\" and .type == \"AAAA\")|.id" /tmp/dns.txt)
    ip=$(jq -r -e ".records | .[] | select(.name == \"$name\" and .type == \"AAAA\")|.value" /tmp/dns.txt)
  else
    ip=$(jsonfilter -i /tmp/dns.txt -e "@.records[@.name='$name'&&@.type='AAAA'].value")
    id=$(jsonfilter -i /tmp/dns.txt -e "@.records[@.name='$name'&&@.type='AAAA'].id")
  fi
elif [[ $record_type == "A" ]]; then
  if [[ $json == "jq" ]]; then
    id=$(jq -r -e ".records | .[] | select(.name == \"$name\" and .type == \"A\")|.id" /tmp/dns.txt)
    ip=$(jq -r -e ".records | .[] | select(.name == \"$name\" and .type == \"A\")|.value" /tmp/dns.txt)
  else
    ip=$(jsonfilter -i /tmp/dns.txt -e "@.records[@.name='$name'&&@.type='A'].value")
    id=$(jsonfilter -i /tmp/dns.txt -e "@.records[@.name='$name'&&@.type='A'].id")
  fi
fi
#echo $name;
#echo $id;
#echo $ip;
#echo $new_ip;
if [[ $ip == $new_ip ]]; then
  echo 'no update needed'
  exit
fi
if [[ $id ]]; then
  echo "mod ip"
  if curl -s -d "login_token=$token&format=json&domain=$domain&record_id=$id&value=$new_ip&record_type=$record_type&record_line_id=0&sub_domain=$name" https://dnsapi.cn/Record.Modify | grep -q '"code":"1"'; then
    echo "sus"
  fi
else
  echo "add ip"
  if curl -s -d "login_token=$token&format=json&domain=$domain&sub_domain=$name&record_type=$record_type&record_line_id=0&value=$new_ip" https://dnsapi.cn/Record.Create | grep -q '"code":"1"'; then
    echo "sus"
  fi
fi

在/lib/netifd/ppp-up文件内调用上面的脚本,当pppoe网络连接成功时会执行此文件,$4变量为pppoe连接的本地IP。

/usr/bin/update-ip.sh name $4 > /dev/null 2>&1 &

pppoe只能传递公网ipv4,使用ifstatus可获取pppoe接口ipv6地址和分配内网的ipv6前缀,根据mac生成的ipv6后缀可为内网其它机器做ddns。

ifstatus wan_6
ifstatus wan
ubus call network.interface dump
jsonfilter -i /tmp/wan6.txt -e '@["ipv6-prefix"][0].address'
jsonfilter -i /tmp/wan6.txt -e '@["ipv6-address"][0].address' 

PHP版本:
https://www.haiyun.me/archives/1186.html

ubuntu 18.04安装php7.2 leveldb

发布时间:January 7, 2019 // 分类: // No Comments

安装:

apt install php7.2-dev libleveldb-dev
git clone https://github.com/reeze/php-leveldb.git
cd php-leveldb/
phpize
./configure 
#arm64
#./configure --with-leveldb=/usr/lib/aarch64-linux-gnu/ 
make && make install
echo 'extension=leveldb.so' > /etc/php/7.2/cli/conf.d/40-leveldb.ini

查看是否加载:

php -m|grep leveldb

unbound过滤ipv6域名查询

发布时间:January 6, 2019 // 分类: // No Comments

将所有的ipv6地址设置为private-address,这样unbound转发域名有ipv6结果时不返回给查询的客户端。
如果对单独的域名进行过滤新建一个server实例,然后forward-zone转发特定域名到过滤ipv6的端口。

server:
 port: 5350
 do-ip4: yes
 do-ip6: no 
 private-address: ::/0
 interface: 0.0.0.0
 access-control: 127.0.0.0/8 allow
 access-control: 192.168.0.0/16 allow
 msg-cache-size: 4m
 rrset-cache-size: 4m
 cache-max-ttl: 3600
 cache-min-ttl: 300
 hide-identity: yes
 hide-version: yes
 prefetch: yes
 num-threads: 4
 do-not-query-localhost: no
#minimal-responses: yes
#qname-minimisation: yes
#tcp-upstream: yes
#verbosity: 1
#logfile: "/var/log/unbound.log"

forward-zone:
  name: "."
  forward-addr: 114.114.114.114

prefetch当查询时且缓存离过期时间还有10%时预请求并更新dns缓存,如果期间无查询请求不更新。
参考:
https://lost-and-found-narihiro.blogspot.com/2011/10/unbound-prefetch.html
https://nlnetlabs.nl/pipermail/unbound-users/2018-January/010444.html
https://calomel.org/unbound_dns.html
https://nlnetlabs.nl/documentation/unbound/unbound.conf/
https://www.nlnetlabs.nl/svn/unbound/trunk/doc/example.conf.in

windows10设置ipv4优先于ipv6

发布时间:January 5, 2019 // 分类: // No Comments

类似于linux下设置ipv4优先,将::ffff:0:0/96的优先级调高即可。
查看优先级顺序:

netsh interface ipv6 show prefixpolicies

2019-01-05_082456.jpg
修改优先级:

netsh interface ipv6 set prefixpolicy ::ffff:0:0/96 100 4

查看修改后的优先级:
2019-01-05_082543.jpg

更新:此方法有问题,重启电脑后ipv6优先级不正常,暂不使用。
参考:
http://blog.ilc.edu.tw/blog/index.php?op=printView&articleId=622435&blogId=25793

transmission rpc api php

发布时间:December 29, 2018 // 分类:PT // No Comments

<?php
class transmission {
  public $url;
  public $user;
  public $pass;
  public $session_id = '';
  public $header = array(
    //'User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:62.0) Gecko/20100101 Firefox/62.0',
    'Connection: keep-alive'
  );

  public function __construct($url, $user, $pass) {
    $this->url = $url;
    $this->user = $user;
    $this->pass = $pass;
    $this->request();
  }

  function request($post = "") {
    //echo $url.PHP_EOL;
    $url = $this->url;
    $curl = curl_init();
    curl_setopt($curl, CURLOPT_URL, $url);
    curl_setopt($curl, CURLOPT_HEADER, 1);
    curl_setopt($curl, CURLOPT_TIMEOUT, 10);
    curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
    curl_setopt($curl, CURLOPT_FOLLOWLOCATION, 1);
    curl_setopt($curl, CURLOPT_MAXREDIRS, 5);
    curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, 0);
    $header = $this->header;
    if ($this->session_id) {
      $header[] = 'X-Transmission-Session-Id: '.$this->session_id;
    }
    $header[] = "Authorization: Basic ".base64_encode($this->user.':'.$this->pass);
    curl_setopt($curl, CURLOPT_HTTPHEADER, $header);
    if ($post) {
      echo $post.PHP_EOL;
      curl_setopt($curl, CURLOPT_POST, 1);
      curl_setopt($curl, CURLOPT_POSTFIELDS, $post);
    }
    $data = curl_exec($curl);
    $code = curl_getinfo($curl)["http_code"];
    $header_size = curl_getinfo($curl, CURLINFO_HEADER_SIZE);
    //var_dump(curl_getinfo($curl));
    curl_close($curl);
    $header = substr($data, 0, $header_size);
    $body = substr($data, $header_size);
    //var_dump($header);
    //var_dump($body);
    if ($code == 409 && preg_match('/<code>X-Transmission-Session-Id: (.*?)<\/code>/', $data, $mh)) {
      $this->session_id = $mh[1];
    } elseif ($code == 401) {
      die("Authorization error".PHP_EOL);
    } elseif ($code == 200) {
      return json_decode($body, 1);
    }
  }

  function build_request($method, $arguments) {
    $post_data = json_encode(array('method' => $method, 'arguments' => $arguments));
    return $this->request($post_data);
  }

  public function get($ids = null, $fields = null) {
    $fields = $fields ? $fields : array( "id", "name", "status", "doneDate", "haveValid", "totalSize", "trackers", "uploadLimited", "uploadLimit");
    $arguments = $ids ? array("fields" => $fields, "ids" => $ids) : array("fields" => $fields);
    return $this->build_request( "torrent-get", $arguments );
  }

  public function reannounce($ids = null) {
    $arguments = $ids ? array("ids" => $ids) : array();
    return $this->build_request( "torrent-reannounce", $arguments);
  }
}

/*
$rpc = new transmission("http://192.168.168.6:9091/transmission/rpc", "admin", "pass");
$data = $rpc->get(array(1), array('id'));
//$data = $rpc->get();
$data = $rpc->reannounce();
var_dump($data);
 */

更多参数:
https://github.com/transmission/transmission/blob/master/extras/rpc-spec.txt

分类
最新文章
最近回复
  • opnfense: 谢谢博主!!!解决问题了!!!我之前一直以为内置的odhcp6就是唯一管理ipv6的方式
  • liyk: 这个方法获取的IPv6大概20分钟之后就会失效,默认路由先消失,然后Global IPV6再消失
  • 海运: 不好意思,没有。
  • zongboa: 您好,請問一下有immortalwrt設定guest Wi-Fi的GUI教學嗎?感謝您。
  • 海运: 恩山有很多。
  • swsend: 大佬可以分享一下固件吗,谢谢。
  • Jimmy: 方法一 nghtp3步骤需要改成如下才能编译成功: git clone https://git...
  • 海运: 地址格式和udpxy一样,udpxy和msd_lite能用这个就能用。
  • 1: 怎么用 编译后的程序在家里路由器内任意一台设备上运行就可以吗?比如笔记本电脑 m参数是笔记本的...
  • 孤狼: ups_status_set: seems that UPS [BK650M2-CH] is ...
归档