summaryrefslogtreecommitdiffhomepage
path: root/dnscompress
diff options
context:
space:
mode:
authorPablo Mazzini <pmazzini@gmail.com>2018-08-01 18:25:54 +0200
committerPablo Mazzini <pmazzini@gmail.com>2018-08-01 18:25:54 +0200
commit833497ba74dfdd0188176ae38a19de7a03a7858a (patch)
tree4422669864704455d6c1e22128629368ab48f384 /dnscompress
parent1f58265bc02e94b0c530719cda87da780a3813c1 (diff)
extract rfc1035label
Diffstat (limited to 'dnscompress')
-rw-r--r--dnscompress/label.go60
-rw-r--r--dnscompress/label_test.go66
2 files changed, 126 insertions, 0 deletions
diff --git a/dnscompress/label.go b/dnscompress/label.go
new file mode 100644
index 0000000..3d84708
--- /dev/null
+++ b/dnscompress/label.go
@@ -0,0 +1,60 @@
+package dnscompress
+
+import (
+ "fmt"
+ "strings"
+)
+
+// This implements the compression from RFC 1035, section 4.1.4
+// https://tools.ietf.org/html/rfc1035
+
+// LabelsFromBytes decodes a serialized stream and returns a list of labels
+func LabelsFromBytes(buf []byte) ([]string, error) {
+ var (
+ pos = 0
+ domains = make([]string, 0)
+ label = ""
+ )
+ for {
+ if pos >= len(buf) {
+ return domains, nil
+ }
+ length := int(buf[pos])
+ pos++
+ if length == 0 {
+ domains = append(domains, label)
+ label = ""
+ }
+ if len(buf)-pos < length {
+ return nil, fmt.Errorf("DomainNamesFromBytes: invalid short label length")
+ }
+ if label != "" {
+ label += "."
+ }
+ label += string(buf[pos : pos+length])
+ pos += length
+ }
+}
+
+// LabelToBytes encodes a label and returns a serialized stream of bytes
+func LabelToBytes(label string) []byte {
+ var encodedLabel []byte
+ if len(label) == 0 {
+ return []byte{0}
+ }
+ for _, part := range strings.Split(label, ".") {
+ encodedLabel = append(encodedLabel, byte(len(part)))
+ encodedLabel = append(encodedLabel, []byte(part)...)
+ }
+ return append(encodedLabel, 0)
+}
+
+// LabelsToBytes encodes a list of labels and returns a serialized stream of
+// bytes
+func LabelsToBytes(labels []string) []byte {
+ var encodedLabels []byte
+ for _, label := range labels {
+ encodedLabels = append(encodedLabels, LabelToBytes(label)...)
+ }
+ return encodedLabels
+}
diff --git a/dnscompress/label_test.go b/dnscompress/label_test.go
new file mode 100644
index 0000000..af8e405
--- /dev/null
+++ b/dnscompress/label_test.go
@@ -0,0 +1,66 @@
+package dnscompress
+
+import (
+ "bytes"
+ "testing"
+)
+
+func TestLabelsFromBytes(t *testing.T) {
+ labels, err := LabelsFromBytes([]byte{
+ 0x9, 's', 'l', 'a', 'c', 'k', 'w', 'a', 'r', 'e',
+ 0x2, 'i', 't',
+ 0x0,
+ })
+ if err != nil {
+ t.Fatal(err)
+ }
+ if len(labels) != 1 {
+ t.Fatalf("Invalid labels length. Expected: 1, got: %v", len(labels))
+ }
+ if labels[0] != "slackware.it" {
+ t.Fatalf("Invalid label. Expected: %v, got: %v'", "slackware.it", labels[0])
+ }
+}
+
+func TestLabelsFromBytesZeroLength(t *testing.T) {
+ labels, err := LabelsFromBytes([]byte{})
+ if err != nil {
+ t.Fatal(err)
+ }
+ if len(labels) != 0 {
+ t.Fatalf("Invalid labels length. Expected: 0, got: %v", len(labels))
+ }
+}
+
+func TestLabelsFromBytesInvalidLength(t *testing.T) {
+ labels, err := LabelsFromBytes([]byte{0x3, 0xaa, 0xbb}) // short length
+ if err == nil {
+ t.Fatal("Expected error, got nil")
+ }
+ if len(labels) != 0 {
+ t.Fatalf("Invalid labels length. Expected: 0, got: %v", len(labels))
+ }
+ if labels != nil {
+ t.Fatalf("Invalid label. Expected nil, got %v", labels)
+ }
+}
+
+func TestLabelToBytes(t *testing.T) {
+ encodedLabel := LabelToBytes("slackware.it")
+ expected := []byte{
+ 0x9, 's', 'l', 'a', 'c', 'k', 'w', 'a', 'r', 'e',
+ 0x2, 'i', 't',
+ 0x0,
+ }
+ if !bytes.Equal(encodedLabel, expected) {
+ t.Fatalf("Invalid label. Expected: %v, got: %v", expected, encodedLabel)
+ }
+}
+
+func TestLabelToBytesZeroLength(t *testing.T) {
+ encodedLabel := LabelToBytes("")
+ expected := []byte{0}
+ if !bytes.Equal(encodedLabel, expected) {
+ t.Fatalf("Invalid label. Expected: %v, got: %v", expected, encodedLabel)
+ }
+}