Skip to content

Commit a8cf7ba

Browse files
add ModifyWithResult to v3
1 parent 5ecadec commit a8cf7ba

File tree

2 files changed

+67
-0
lines changed

2 files changed

+67
-0
lines changed

v3/client.go

+1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ type Client interface {
2121
Del(*DelRequest) error
2222
Modify(*ModifyRequest) error
2323
ModifyDN(*ModifyDNRequest) error
24+
ModifyWithResult(*ModifyRequest) (*ModifyResult, error)
2425

2526
Compare(dn, attribute, value string) (bool, error)
2627
PasswordModify(*PasswordModifyRequest) (*PasswordModifyResult, error)

v3/modify.go

+66
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@
2626
package ldap
2727

2828
import (
29+
"errors"
2930
"log"
3031

3132
ber "github.com/go-asn1-ber/asn1-ber"
@@ -38,6 +39,12 @@ const (
3839
ReplaceAttribute = 2
3940
)
4041

42+
// ModifyResult holds the server's response to a modify request
43+
type ModifyResult struct {
44+
// Controls are the returned controls
45+
Controls []Control
46+
}
47+
4148
// PartialAttribute for a ModifyRequest as defined in https://tools.ietf.org/html/rfc4511
4249
type PartialAttribute struct {
4350
// Type is the type of the partial attribute
@@ -149,3 +156,62 @@ func (l *Conn) Modify(modifyRequest *ModifyRequest) error {
149156
}
150157
return nil
151158
}
159+
160+
// ModifyWithResult performs the ModifyRequest and returns the result
161+
func (l *Conn) ModifyWithResult(modifyRequest *ModifyRequest) (*ModifyResult, error) {
162+
packet := ber.Encode(ber.ClassUniversal, ber.TypeConstructed, ber.TagSequence, nil, "LDAP Request")
163+
packet.AppendChild(ber.NewInteger(ber.ClassUniversal, ber.TypePrimitive, ber.TagInteger, l.nextMessageID(), "MessageID"))
164+
packet.AppendChild(modifyRequest.encode())
165+
if len(modifyRequest.Controls) > 0 {
166+
packet.AppendChild(encodeControls(modifyRequest.Controls))
167+
}
168+
169+
l.Debug.PrintPacket(packet)
170+
171+
msgCtx, err := l.sendMessage(packet)
172+
if err != nil {
173+
return nil, err
174+
}
175+
defer l.finishMessage(msgCtx)
176+
177+
result := &ModifyResult{
178+
Controls: make([]Control, 0),
179+
}
180+
181+
l.Debug.Printf("%d: waiting for response", msgCtx.id)
182+
packetResponse, ok := <-msgCtx.responses
183+
if !ok {
184+
return nil, NewError(ErrorNetwork, errors.New("ldap: response channel closed"))
185+
}
186+
packet, err = packetResponse.ReadPacket()
187+
l.Debug.Printf("%d: got response %p", msgCtx.id, packet)
188+
if err != nil {
189+
return nil, err
190+
}
191+
192+
if l.Debug {
193+
if err := addLDAPDescriptions(packet); err != nil {
194+
return nil, err
195+
}
196+
ber.PrintPacket(packet)
197+
}
198+
199+
switch packet.Children[1].Tag {
200+
case ApplicationModifyResponse:
201+
err := GetLDAPError(packet)
202+
if err != nil {
203+
return nil, err
204+
}
205+
if len(packet.Children) == 3 {
206+
for _, child := range packet.Children[2].Children {
207+
decodedChild, err := DecodeControl(child)
208+
if err != nil {
209+
return nil, errors.New("failed to decode child control: " + err.Error())
210+
}
211+
result.Controls = append(result.Controls, decodedChild)
212+
}
213+
}
214+
}
215+
l.Debug.Printf("%d: returning", msgCtx.id)
216+
return result, nil
217+
}

0 commit comments

Comments
 (0)