View Javadoc
1   /*
2    *  Licensed to the Apache Software Foundation (ASF) under one
3    *  or more contributor license agreements.  See the NOTICE file
4    *  distributed with this work for additional information
5    *  regarding copyright ownership.  The ASF licenses this file
6    *  to you under the Apache License, Version 2.0 (the
7    *  "License"); you may not use this file except in compliance
8    *  with the License.  You may obtain a copy of the License at
9    *
10   *    https://www.apache.org/licenses/LICENSE-2.0
11   *
12   *  Unless required by applicable law or agreed to in writing,
13   *  software distributed under the License is distributed on an
14   *  "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15   *  KIND, either express or implied.  See the License for the
16   *  specific language governing permissions and limitations
17   *  under the License.
18   *
19   */
20  package org.apache.directory.api.ldap.codec.actions.controls;
21  
22  
23  import org.apache.directory.api.asn1.DecoderException;
24  import org.apache.directory.api.asn1.ber.grammar.GrammarAction;
25  import org.apache.directory.api.asn1.ber.tlv.TLV;
26  import org.apache.directory.api.asn1.util.Oid;
27  import org.apache.directory.api.i18n.I18n;
28  import org.apache.directory.api.ldap.codec.api.ControlFactory;
29  import org.apache.directory.api.ldap.codec.api.LdapApiService;
30  import org.apache.directory.api.ldap.codec.api.LdapMessageContainer;
31  import org.apache.directory.api.ldap.model.message.Control;
32  import org.apache.directory.api.ldap.model.message.Message;
33  import org.apache.directory.api.ldap.model.message.Request;
34  import org.apache.directory.api.ldap.model.message.controls.OpaqueControl;
35  import org.apache.directory.api.util.Strings;
36  import org.slf4j.Logger;
37  import org.slf4j.LoggerFactory;
38  
39  
40  /**
41   * The action used add a new control. We store its OID.
42   * <pre>
43   * Control ::= SEQUENCE {
44   *     controlType             LDAPOID,
45   *     ...LdapMessageContainerDirect&lt;Message&gt;
46   * </pre>
47   *
48   * @author <a href="mailto:dev@directory.apache.org">Apache Directory Project</a>
49   */
50  public class StoreControlName extends GrammarAction<LdapMessageContainer<Message>>
51  {
52      /** The logger */
53      private static final Logger LOG = LoggerFactory.getLogger( StoreControlName.class );
54  
55      /**
56       * Instantiates a new AddControl action.
57       */
58      public StoreControlName()
59      {
60          super( "Add a new control" );
61      }
62  
63  
64      /**
65       * {@inheritDoc}
66       */
67      @Override
68      public void action( LdapMessageContainer<Message> container ) throws DecoderException
69      {
70          TLV tlv = container.getCurrentTLV();
71  
72          // Store the type
73          // We have to handle the special case of a 0 length OID
74          if ( tlv.getLength() == 0 )
75          {
76              String msg = I18n.err( I18n.ERR_08214_NULL_OID );
77              LOG.error( msg );
78  
79              // This will generate a PROTOCOL_ERROR
80              throw new DecoderException( msg );
81          }
82  
83          byte[] value = tlv.getValue().getData();
84          String oidValue = Strings.asciiBytesToString( value );
85  
86          // The OID is encoded as a String, not an Object Id
87          if ( !Oid.isOid( oidValue ) )
88          {
89              String msg = I18n.err( I18n.ERR_08215_INVALID_CONTROL_OID, oidValue );
90              LOG.error( msg );
91  
92              // This will generate a PROTOCOL_ERROR
93              throw new DecoderException( msg );
94          }
95  
96          // Search for the control. It can be a request or a response control.
97          // If the control is not known, we create an Opaque control
98          Message message = container.getMessage();
99          LdapApiService codec = container.getLdapCodecService();
100         ControlFactory<? extends Control> controlFactory;
101         
102         Control control;
103 
104         if ( message instanceof Request )
105         {
106             controlFactory = codec.getRequestControlFactories().get( oidValue );
107         }
108         else
109         {
110             controlFactory = codec.getResponseControlFactories().get( oidValue );
111         }
112 
113         if ( controlFactory == null )
114         {
115             control =  new OpaqueControl( oidValue );
116         }
117         else
118         {
119             control = controlFactory.newControl();
120         }
121 
122         container.setControlFactory( controlFactory );
123         
124         // At this point, the control exists, we may have to decode the value and feed it
125         //In any case, add it to the message's controls, and store it in the container for further
126         // processing (aka, value decoding)
127         message.addControl( control );
128 
129         container.setCurrentControl( control );
130 
131         // We can have an END transition
132         container.setGrammarEndAllowed( true );
133 
134         if ( LOG.isDebugEnabled() )
135         {
136             LOG.debug( I18n.msg( I18n.MSG_08201_CONTROL_OID, oidValue ) );
137         }
138     }
139 }