From cb83d2a3368cad35f592190b550d343d901815f9 Mon Sep 17 00:00:00 2001 From: Christopher Puschmann Date: Mon, 25 Apr 2022 18:51:49 +0200 Subject: [PATCH] Implement Subtree delete control type (#373) --- control.go | 29 +++++++++++++++++++++++++++++ control_test.go | 8 ++++++++ v3/control.go | 29 +++++++++++++++++++++++++++++ v3/control_test.go | 8 ++++++++ 4 files changed, 74 insertions(+) diff --git a/control.go b/control.go index 64fb002a..aa68fc83 100644 --- a/control.go +++ b/control.go @@ -20,6 +20,8 @@ const ( ControlTypeManageDsaIT = "2.16.840.1.113730.3.4.2" // ControlTypeWhoAmI - https://tools.ietf.org/html/rfc4532 ControlTypeWhoAmI = "1.3.6.1.4.1.4203.1.11.3" + // ControlTypeSubTreeDelete - https://datatracker.ietf.org/doc/html/draft-armijo-ldap-treedelete-02 + ControlTypeSubtreeDelete = "1.2.840.113556.1.4.805" // ControlTypeMicrosoftNotification - https://msdn.microsoft.com/en-us/library/aa366983(v=vs.85).aspx ControlTypeMicrosoftNotification = "1.2.840.113556.1.4.528" @@ -34,6 +36,7 @@ var ControlTypeMap = map[string]string{ ControlTypePaging: "Paging", ControlTypeBeheraPasswordPolicy: "Password Policy - Behera Draft", ControlTypeManageDsaIT: "Manage DSA IT", + ControlTypeSubtreeDelete: "Subtree Delete Control", ControlTypeMicrosoftNotification: "Change Notification - Microsoft", ControlTypeMicrosoftShowDeleted: "Show Deleted Objects - Microsoft", ControlTypeMicrosoftServerLinkTTL: "Return TTL-DNs for link values with associated expiry times - Microsoft", @@ -485,6 +488,8 @@ func DecodeControl(packet *ber.Packet) (Control, error) { return NewControlMicrosoftShowDeleted(), nil case ControlTypeMicrosoftServerLinkTTL: return NewControlMicrosoftServerLinkTTL(), nil + case ControlTypeSubtreeDelete: + return NewControlSubtreeDelete(), nil default: c := new(ControlString) c.ControlType = ControlType @@ -519,6 +524,30 @@ func NewControlBeheraPasswordPolicy() *ControlBeheraPasswordPolicy { } } +type ControlSubtreeDelete struct{} + +func (c *ControlSubtreeDelete) GetControlType() string { + return ControlTypeSubtreeDelete +} + +func NewControlSubtreeDelete() *ControlSubtreeDelete { + return &ControlSubtreeDelete{} +} + +func (c *ControlSubtreeDelete) Encode() *ber.Packet { + packet := ber.Encode(ber.ClassUniversal, ber.TypeConstructed, ber.TagSequence, nil, "Control") + packet.AppendChild(ber.NewString(ber.ClassUniversal, ber.TypePrimitive, ber.TagOctetString, ControlTypeSubtreeDelete, "Control Type ("+ControlTypeMap[ControlTypeSubtreeDelete]+")")) + + return packet +} + +func (c *ControlSubtreeDelete) String() string { + return fmt.Sprintf( + "Control Type: %s (%q)", + ControlTypeMap[ControlTypeSubtreeDelete], + ControlTypeSubtreeDelete) +} + func encodeControls(controls []Control) *ber.Packet { packet := ber.Encode(ber.ClassContext, ber.TypeConstructed, 0, nil, "Controls") for _, control := range controls { diff --git a/control_test.go b/control_test.go index db3a35ea..2bdbcec7 100644 --- a/control_test.go +++ b/control_test.go @@ -32,6 +32,10 @@ func TestControlMicrosoftServerLinkTTL(t *testing.T) { runControlTest(t, NewControlMicrosoftServerLinkTTL()) } +func TestControlSubtreeDelete(t *testing.T) { + runControlTest(t, NewControlSubtreeDelete()) +} + func TestControlString(t *testing.T) { runControlTest(t, NewControlString("x", true, "y")) runControlTest(t, NewControlString("x", true, "")) @@ -89,6 +93,10 @@ func TestDescribeControlPaging(t *testing.T) { runAddControlDescriptions(t, NewControlPaging(0), "Control Type (Paging)", "Control Value (Paging)") } +func TestDescribeControlSubtreeDelete(t *testing.T) { + runAddControlDescriptions(t, NewControlSubtreeDelete(), "Control Type (Subtree Delete Control)") +} + func TestDescribeControlMicrosoftNotification(t *testing.T) { runAddControlDescriptions(t, NewControlMicrosoftNotification(), "Control Type (Change Notification - Microsoft)") } diff --git a/v3/control.go b/v3/control.go index 1d442443..5780ddd8 100644 --- a/v3/control.go +++ b/v3/control.go @@ -20,6 +20,8 @@ const ( ControlTypeManageDsaIT = "2.16.840.1.113730.3.4.2" // ControlTypeWhoAmI - https://tools.ietf.org/html/rfc4532 ControlTypeWhoAmI = "1.3.6.1.4.1.4203.1.11.3" + // ControlTypeSubTreeDelete - https://datatracker.ietf.org/doc/html/draft-armijo-ldap-treedelete-02 + ControlTypeSubtreeDelete = "1.2.840.113556.1.4.805" // ControlTypeMicrosoftNotification - https://msdn.microsoft.com/en-us/library/aa366983(v=vs.85).aspx ControlTypeMicrosoftNotification = "1.2.840.113556.1.4.528" @@ -37,6 +39,7 @@ var ControlTypeMap = map[string]string{ ControlTypePaging: "Paging", ControlTypeBeheraPasswordPolicy: "Password Policy - Behera Draft", ControlTypeManageDsaIT: "Manage DSA IT", + ControlTypeSubtreeDelete: "Subtree Delete Control", ControlTypeMicrosoftNotification: "Change Notification - Microsoft", ControlTypeMicrosoftShowDeleted: "Show Deleted Objects - Microsoft", ControlTypeMicrosoftServerLinkTTL: "Return TTL-DNs for link values with associated expiry times - Microsoft", @@ -504,6 +507,8 @@ func DecodeControl(packet *ber.Packet) (Control, error) { return nil, err } return &ControlServerSideSortingResponse{Result: ret, Criticality: Criticality}, nil + case ControlTypeSubtreeDelete: + return NewControlSubtreeDelete(), nil default: c := new(ControlString) c.ControlType = ControlType @@ -538,6 +543,30 @@ func NewControlBeheraPasswordPolicy() *ControlBeheraPasswordPolicy { } } +type ControlSubtreeDelete struct{} + +func (c *ControlSubtreeDelete) GetControlType() string { + return ControlTypeSubtreeDelete +} + +func NewControlSubtreeDelete() *ControlSubtreeDelete { + return &ControlSubtreeDelete{} +} + +func (c *ControlSubtreeDelete) Encode() *ber.Packet { + packet := ber.Encode(ber.ClassUniversal, ber.TypeConstructed, ber.TagSequence, nil, "Control") + packet.AppendChild(ber.NewString(ber.ClassUniversal, ber.TypePrimitive, ber.TagOctetString, ControlTypeSubtreeDelete, "Control Type ("+ControlTypeMap[ControlTypeSubtreeDelete]+")")) + + return packet +} + +func (c *ControlSubtreeDelete) String() string { + return fmt.Sprintf( + "Control Type: %s (%q)", + ControlTypeMap[ControlTypeSubtreeDelete], + ControlTypeSubtreeDelete) +} + func encodeControls(controls []Control) *ber.Packet { packet := ber.Encode(ber.ClassContext, ber.TypeConstructed, 0, nil, "Controls") for _, control := range controls { diff --git a/v3/control_test.go b/v3/control_test.go index db3a35ea..2bdbcec7 100644 --- a/v3/control_test.go +++ b/v3/control_test.go @@ -32,6 +32,10 @@ func TestControlMicrosoftServerLinkTTL(t *testing.T) { runControlTest(t, NewControlMicrosoftServerLinkTTL()) } +func TestControlSubtreeDelete(t *testing.T) { + runControlTest(t, NewControlSubtreeDelete()) +} + func TestControlString(t *testing.T) { runControlTest(t, NewControlString("x", true, "y")) runControlTest(t, NewControlString("x", true, "")) @@ -89,6 +93,10 @@ func TestDescribeControlPaging(t *testing.T) { runAddControlDescriptions(t, NewControlPaging(0), "Control Type (Paging)", "Control Value (Paging)") } +func TestDescribeControlSubtreeDelete(t *testing.T) { + runAddControlDescriptions(t, NewControlSubtreeDelete(), "Control Type (Subtree Delete Control)") +} + func TestDescribeControlMicrosoftNotification(t *testing.T) { runAddControlDescriptions(t, NewControlMicrosoftNotification(), "Control Type (Change Notification - Microsoft)") }