;;;; Copyright (C) 2021 Julien Lepiller ;;;; ;;;; This library is free software; you can redistribute it and/or ;;;; modify it under the terms of the GNU Lesser General Public ;;;; License as published by the Free Software Foundation; either ;;;; version 3 of the License, or (at your option) any later version. ;;;; ;;;; This library is distributed in the hope that it will be useful, ;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of ;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU ;;;; Lesser General Public License for more details. ;;;; ;;;; You should have received a copy of the GNU Lesser General Public ;;;; License along with this library; if not, write to the Free Software ;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA ;;;; (define-module (ip addr) #:use-module (ice-9 match) #:use-module (ip link) #:use-module (ip utils) #:use-module (netlink route addr) #:use-module (netlink route attrs) #:use-module (netlink connection) #:use-module (netlink constant) #:use-module (netlink deserialize) #:use-module (netlink message) #:use-module (netlink standard) #:use-module (srfi srfi-1) #:use-module (srfi srfi-9) #:export (addr-add addr-del addr-show)) (define-record-type (make-addr family prefix flags scope link label addr brd cacheinfo) addr? (family addr-family) (prefix addr-prefix) (flags addr-flags) (scope addr-scope) (link addr-link) (label addr-label) (addr addr-addr) (brd addr-brd) (cacheinfo addr-cacheinfo)) (define (cidr->addr str) (match (string-split str #\/) ((addr) addr) ((addr prefix) addr) (_ (throw 'incorrect-cidr-notation str)))) (define (cidr->prefix str) (match (string-split str #\/) ((addr) #f) ((addr prefix) (string->number prefix)) (_ (throw 'incorrect-cidr-notation str)))) (define* (addr-del device cidr #:key (ipv6? #f)) (define request-num (random 65535)) (define prefix (cidr->prefix cidr)) (define addr (cidr->addr cidr)) (define index (cond ((number? device) device) ((string? device) (link-name->index device)))) (define message (make-message RTM_DELADDR (logior NLM_F_REQUEST NLM_F_ACK) request-num 0 (make-addr-message (if ipv6? AF_INET6 AF_INET) (if prefix prefix 0) 0 0 index (list (make-route-attr IFA_LOCAL ((if ipv6? make-ipv6-route-attr make-ipv4-route-attr) addr)) (make-route-attr IFA_ADDRESS ((if ipv6? make-ipv6-route-attr make-ipv4-route-attr) addr)))))) (let ((sock (connect-route))) (send-msg message sock) (let ((answer (receive-and-decode-msg sock %default-route-decoder))) (close-socket sock) (answer-ok? (last answer))))) (define* (addr-add device cidr #:key (ipv6? #f)) (define request-num (random 65535)) (define prefix (cidr->prefix cidr)) (define addr (cidr->addr cidr)) (define index (cond ((number? device) device) ((string? device) (link-name->index device)))) (define message (make-message RTM_NEWADDR (logior NLM_F_REQUEST NLM_F_ACK NLM_F_EXCL NLM_F_CREATE) request-num 0 (make-addr-message (if ipv6? AF_INET6 AF_INET) (if prefix prefix 0) 0 0 index (list (make-route-attr IFA_LOCAL ((if ipv6? make-ipv6-route-attr make-ipv4-route-attr) addr)) (make-route-attr IFA_ADDRESS ((if ipv6? make-ipv6-route-attr make-ipv4-route-attr) addr)))))) (let ((sock (connect-route))) (send-msg message sock) (let ((answer (receive-and-decode-msg sock %default-route-decoder))) (close-socket sock) (answer-ok? (last answer))))) (define (get-addrs) (define request-num (random 65535)) (define message (make-message RTM_GETADDR (logior NLM_F_REQUEST NLM_F_DUMP) request-num 0 (make-addr-message AF_UNSPEC 0 0 0 0 '()))) (let ((sock (connect-route))) (send-msg message sock) (let* ((answer (receive-and-decode-msg sock %default-route-decoder)) (addrs (filter (lambda (msg) (equal? (message-kind msg) RTM_NEWADDR)) answer)) (addrs (map (lambda (msg) (let* ((data (message-data msg)) (attrs (addr-message-attrs data))) (make-addr (addr-message-family data) (addr-message-prefix-len data) (map int->ifa-flag (split-flags (logior (addr-message-flags data) (get-attr attrs IFA_FLAGS)))) (addr-message-scope data) (addr-message-index data) (get-attr attrs IFA_LABEL) (get-attr attrs IFA_ADDRESS) (get-attr attrs IFA_BROADCAST) (get-attr attrs IFA_CACHEINFO)))) addrs))) (close-socket sock) addrs))) (define print-addr (match-lambda (($ family prefix flags scope link label addr brd cacheinfo) (format #t " ~a ~a/~a" (cond ((= family AF_INET) "inet") ((= family AF_INET6) "inet6") (else "????")) addr prefix) (when brd (format #t " brd ~a" brd)) (when scope (format #t " scope ~a" (cond ((equal? scope RT_SCOPE_UNIVERSE) "global") (else (string-downcase (substring (symbol->string (int->rtm-scope scope)) 9)))))) (for-each (lambda (flag) (unless (equal? flag 'IFA_F_PERMANENT) (format #t " ~a" (string-downcase (substring (symbol->string flag) 6))))) flags) (when label (format #t " ~a" label)) (format #t "~%") (when cacheinfo (if (member 'IFA_F_PERMANENT flags) (format #t " valid_lft forever preferred_lft forever~%") (format #t " valid_lft ~asec preferred_lft ~asec~%" (route-cache-info-attr-type-valid cacheinfo) (route-cache-info-attr-type-prefered cacheinfo))))))) (define* (addr-show #:optional (device #f)) (define links (get-links)) (define index (cond ((number? device) device) ((string? device) (link-name->index device)) (else #f))) (define addrs (get-addrs)) (for-each (lambda (link) (unless (and index (not (equal? (link-id link) index))) (print-link link) (for-each print-addr (filter (lambda (addr) (equal? (link-id link) (addr-link addr))) addrs)))) links))