guile-netlink/ip/addr.scm

addr.scm

1
;;;; Copyright (C) 2021 Julien Lepiller <julien@lepiller.eu>
2
;;;; 
3
;;;; This library is free software; you can redistribute it and/or
4
;;;; modify it under the terms of the GNU Lesser General Public
5
;;;; License as published by the Free Software Foundation; either
6
;;;; version 3 of the License, or (at your option) any later version.
7
;;;; 
8
;;;; This library is distributed in the hope that it will be useful,
9
;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
10
;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
11
;;;; Lesser General Public License for more details.
12
;;;; 
13
;;;; You should have received a copy of the GNU Lesser General Public
14
;;;; License along with this library; if not, write to the Free Software
15
;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
16
;;;; 
17
18
(define-module (ip addr)
19
  #:use-module (ice-9 match)
20
  #:use-module (ip link)
21
  #:use-module (ip utils)
22
  #:use-module (netlink route addr)
23
  #:use-module (netlink route attrs)
24
  #:use-module (netlink connection)
25
  #:use-module (netlink constant)
26
  #:use-module (netlink deserialize)
27
  #:use-module (netlink message)
28
  #:use-module (netlink standard)
29
  #:use-module (srfi srfi-1)
30
  #:export (addr-add
31
            addr-del
32
            addr-show))
33
34
(define (cidr->addr str)
35
  (match (string-split str #\/)
36
    ((addr) addr)
37
    ((addr prefix) addr)
38
    (_ (throw 'incorrect-cidr-notation str))))
39
40
(define (cidr->prefix str)
41
  (match (string-split str #\/)
42
    ((addr) #f)
43
    ((addr prefix) (string->number prefix))
44
    (_ (throw 'incorrect-cidr-notation str))))
45
46
(define* (addr-del device cidr #:key (ipv6? #f))
47
  (define request-num (random 65535))
48
  (define prefix (cidr->prefix cidr))
49
  (define addr (cidr->addr cidr))
50
51
  (define index
52
    (cond
53
      ((number? device) device)
54
      ((string? device) (link-name->index device))))
55
56
  (define message
57
    (make-message
58
      RTM_DELADDR
59
      (logior NLM_F_REQUEST NLM_F_ACK)
60
      request-num
61
      0
62
      (make-addr-message
63
        (if ipv6? AF_INET6 AF_INET)
64
        (if prefix prefix 0)
65
        0
66
        0
67
        index
68
        (list
69
          (make-route-attr IFA_LOCAL
70
            ((if ipv6?
71
                 make-ipv6-route-attr
72
                 make-ipv4-route-attr)
73
             addr))
74
          (make-route-attr IFA_ADDRESS
75
            ((if ipv6?
76
                 make-ipv6-route-attr
77
                 make-ipv4-route-attr)
78
             addr))))))
79
80
  (let ((sock (connect-route)))
81
    (send-msg message sock)
82
    (let ((answer (receive-and-decode-msg sock %default-route-decoder)))
83
      (close-socket sock)
84
      (answer-ok? (last answer)))))
85
86
(define* (addr-add device cidr #:key (ipv6? #f))
87
  (define request-num (random 65535))
88
  (define prefix (cidr->prefix cidr))
89
  (define addr (cidr->addr cidr))
90
91
  (define index
92
    (cond
93
      ((number? device) device)
94
      ((string? device) (link-name->index device))))
95
96
  (define message
97
    (make-message
98
      RTM_NEWADDR
99
      (logior NLM_F_REQUEST NLM_F_ACK NLM_F_EXCL NLM_F_CREATE)
100
      request-num
101
      0
102
      (make-addr-message
103
        (if ipv6? AF_INET6 AF_INET)
104
        (if prefix prefix 0)
105
        0
106
        0
107
        index
108
        (list
109
          (make-route-attr IFA_LOCAL
110
            ((if ipv6?
111
                 make-ipv6-route-attr
112
                 make-ipv4-route-attr)
113
             addr))
114
          (make-route-attr IFA_ADDRESS
115
            ((if ipv6?
116
                 make-ipv6-route-attr
117
                 make-ipv4-route-attr)
118
             addr))))))
119
120
  (let ((sock (connect-route)))
121
    (send-msg message sock)
122
    (let ((answer (receive-and-decode-msg sock %default-route-decoder)))
123
      (close-socket sock)
124
      (answer-ok? (last answer)))))
125