gssapi_mechanism_base.cpp 8.78 KB
Newer Older
1
/*
2
    Copyright (c) 2007-2014 Contributors as noted in the AUTHORS file
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20

    This file is part of 0MQ.

    0MQ 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.

    0MQ 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 program.  If not, see <http://www.gnu.org/licenses/>.
*/

#include "platform.hpp"
Chris Laws's avatar
Chris Laws committed
21 22 23

#ifdef HAVE_LIBGSSAPI_KRB5

24 25 26 27 28 29 30 31 32 33 34 35 36
#ifdef ZMQ_HAVE_WINDOWS
#include "windows.hpp"
#endif

#include <string.h>
#include <string>

#include "msg.hpp"
#include "session_base.hpp"
#include "err.hpp"
#include "gssapi_mechanism_base.hpp"
#include "wire.hpp"

37 38
zmq::gssapi_mechanism_base_t::gssapi_mechanism_base_t (const options_t & options_) :
    mechanism_t(options_),
39 40
    send_tok (),
    recv_tok (),
41
    /// FIXME remove? in_buf (),
42
    target_name (GSS_C_NO_NAME),
Chris Busbey's avatar
Chris Busbey committed
43
    principal_name (NULL),
Chris Busbey's avatar
Chris Busbey committed
44
    maj_stat (GSS_S_COMPLETE),
45 46 47 48 49
    min_stat (0),
    init_sec_min_stat (0),
    ret_flags (0),
    gss_flags (GSS_C_MUTUAL_FLAG | GSS_C_REPLAY_FLAG),
    cred (GSS_C_NO_CREDENTIAL),
50 51
    context (GSS_C_NO_CONTEXT),
    do_encryption (!options_.gss_plaintext)
52 53 54 55 56
{
}

zmq::gssapi_mechanism_base_t::~gssapi_mechanism_base_t ()
{
57 58 59 60
    if(target_name)
        gss_release_name(&min_stat, &target_name);
    if(context)
        gss_delete_sec_context(&min_stat, &context, GSS_C_NO_BUFFER);
61 62
}

63
int zmq::gssapi_mechanism_base_t::encode_message (msg_t *msg_)
64
{
65 66 67 68
    // Wrap the token value
    int state;
    gss_buffer_desc plaintext;
    gss_buffer_desc wrapped;
69 70 71 72 73 74 75 76 77 78 79

    uint8_t flags = 0;
    if (msg_->flags () & msg_t::more)
        flags |= 0x01;

    uint8_t *plaintext_buffer = static_cast <uint8_t *>(malloc(msg_->size ()+1));
    plaintext_buffer[0] = flags;
    memcpy (plaintext_buffer+1, msg_->data(), msg_->size());

    plaintext.value = plaintext_buffer;
    plaintext.length = msg_->size ()+1;
Chris Laws's avatar
Chris Laws committed
80

81 82
    maj_stat = gss_wrap(&min_stat, context, 1, GSS_C_QOP_DEFAULT,
                        &plaintext, &state, &wrapped);
Chris Laws's avatar
Chris Laws committed
83

84 85
    zmq_assert (maj_stat == GSS_S_COMPLETE);
    zmq_assert (state);
86

87 88 89
    // Re-initialize msg_ for wrapped text
    int rc = msg_->close ();
    zmq_assert (rc == 0);
90

91 92
    rc = msg_->init_size (8 + 4 + wrapped.length);
    zmq_assert (rc == 0);
93

94
    uint8_t *ptr = static_cast <uint8_t *> (msg_->data ());
Chris Laws's avatar
Chris Laws committed
95

96 97 98
    // Add command string
    memcpy (ptr, "\x07MESSAGE", 8);
    ptr += 8;
99 100

    // Add token length
101
    put_uint32 (ptr, static_cast <uint32_t> (wrapped.length));
102 103
    ptr += 4;

104 105 106
    // Add wrapped token value
    memcpy (ptr, wrapped.value, wrapped.length);
    ptr += wrapped.length;
107

108
    gss_release_buffer (&min_stat, &wrapped);
109 110 111 112

    return 0;
}

113
int zmq::gssapi_mechanism_base_t::decode_message (msg_t *msg_)
114
{
115
    const uint8_t *ptr = static_cast <uint8_t *> (msg_->data ());
116 117
    size_t bytes_left = msg_->size ();

118 119
    // Get command string
    if (bytes_left < 8 || memcmp (ptr, "\x07MESSAGE", 8)) {
120 121 122
        errno = EPROTO;
        return -1;
    }
123 124
    ptr += 8;
    bytes_left -= 8;
125 126 127 128 129 130

    // Get token length
    if (bytes_left < 4) {
        errno = EPROTO;
        return -1;
    }
131 132
    gss_buffer_desc wrapped;
    wrapped.length = get_uint32 (ptr);
133 134
    ptr += 4;
    bytes_left -= 4;
Chris Laws's avatar
Chris Laws committed
135

136 137
    // Get token value
    if (bytes_left < wrapped.length) {
138 139 140
        errno = EPROTO;
        return -1;
    }
141 142 143 144 145 146 147 148
    // TODO: instead of malloc/memcpy, can we just do: wrapped.value = ptr;
    const size_t alloc_length = wrapped.length? wrapped.length: 1;
    wrapped.value = static_cast <char *> (malloc (alloc_length));
    if (wrapped.length) {
        alloc_assert (wrapped.value);
        memcpy(wrapped.value, ptr, wrapped.length);
        ptr += wrapped.length;
        bytes_left -= wrapped.length;
149 150
    }

151 152 153 154 155 156 157 158 159 160 161 162 163
    // Unwrap the token value
    int state;
    gss_buffer_desc plaintext;
    maj_stat = gss_unwrap(&min_stat, context, &wrapped, &plaintext,
                          &state, (gss_qop_t *) NULL);

    zmq_assert(maj_stat == GSS_S_COMPLETE);
    zmq_assert(state);

    // Re-initialize msg_ for plaintext
    int rc = msg_->close ();
    zmq_assert (rc == 0);

164
    rc = msg_->init_size (plaintext.length-1);
165
    zmq_assert (rc == 0);
166 167 168 169

    const uint8_t flags = static_cast <char *> (plaintext.value)[0];
    if (flags & 0x01)
	    msg_->set_flags (msg_t::more);
Chris Laws's avatar
Chris Laws committed
170

171
    memcpy (msg_->data (), static_cast <char *> (plaintext.value)+1, plaintext.length-1);
Chris Laws's avatar
Chris Laws committed
172

173 174 175
    gss_release_buffer (&min_stat, &plaintext);
    gss_release_buffer (&min_stat, &wrapped);

176 177 178 179
    if (bytes_left > 0) {
        errno = EPROTO;
        return -1;
    }
180 181 182 183

    return 0;
}

184
int zmq::gssapi_mechanism_base_t::produce_initiate (msg_t *msg_, void *token_value_, size_t token_length_)
185
{
186 187
    zmq_assert (token_value_);
    zmq_assert (token_length_ <= 0xFFFFFFFFUL);
188

189
    const size_t command_size = 9 + 4 + token_length_;
Chris Laws's avatar
Chris Laws committed
190

191 192
    const int rc = msg_->init_size (command_size);
    errno_assert (rc == 0);
Chris Laws's avatar
Chris Laws committed
193

194
    uint8_t *ptr = static_cast <uint8_t *> (msg_->data ());
Chris Laws's avatar
Chris Laws committed
195

196 197 198 199 200 201 202 203 204 205 206 207
    // Add command string
    memcpy (ptr, "\x08INITIATE", 9);
    ptr += 9;

    // Add token length
    put_uint32 (ptr, static_cast <uint32_t> (token_length_));
    ptr += 4;

    // Add token value
    memcpy (ptr, token_value_, token_length_);
    ptr += token_length_;

208 209 210
    return 0;
}

211
int zmq::gssapi_mechanism_base_t::process_initiate (msg_t *msg_, void **token_value_, size_t &token_length_)
212
{
213
    zmq_assert (token_value_);
Chris Laws's avatar
Chris Laws committed
214

215 216
    const uint8_t *ptr = static_cast <uint8_t *> (msg_->data ());
    size_t bytes_left = msg_->size ();
217

218 219 220 221 222 223 224
    // Get command string
    if (bytes_left < 9 || memcmp (ptr, "\x08INITIATE", 9)) {
        errno = EPROTO;
        return -1;
    }
    ptr += 9;
    bytes_left -= 9;
225

226 227 228 229 230 231 232 233
    // Get token length
    if (bytes_left < 4) {
        errno = EPROTO;
        return -1;
    }
    token_length_ = get_uint32 (ptr);
    ptr += 4;
    bytes_left -= 4;
Chris Laws's avatar
Chris Laws committed
234

235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251
    // Get token value
    if (bytes_left < token_length_) {
        errno = EPROTO;
        return -1;
    }
    *token_value_ = static_cast <char *> (malloc (token_length_ ? token_length_ : 1));
    if (token_length_) {
        alloc_assert (*token_value_);
        memcpy(*token_value_, ptr, token_length_);
        ptr += token_length_;
        bytes_left -= token_length_;
    }

    if (bytes_left > 0) {
        errno = EPROTO;
        return -1;
    }
Chris Laws's avatar
Chris Laws committed
252

253 254 255
    return 0;
}

256
int zmq::gssapi_mechanism_base_t::produce_ready (msg_t *msg_)
257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273
{
    unsigned char * const command_buffer = (unsigned char *) malloc (512);
    alloc_assert (command_buffer);

    unsigned char *ptr = command_buffer;

    //  Add command name
    memcpy (ptr, "\x05READY", 6);
    ptr += 6;

    //  Add socket type property
    const char *socket_type = socket_type_string (options.type);
    ptr += add_property (ptr, "Socket-Type", socket_type, strlen (socket_type));

    //  Add identity property
    if (options.type == ZMQ_REQ
    ||  options.type == ZMQ_DEALER
Pieter Hintjens's avatar
Pieter Hintjens committed
274 275
    ||  options.type == ZMQ_ROUTER)
        ptr += add_property (ptr, "Identity", options.identity, options.identity_size);
276 277 278 279 280 281 282

    const size_t command_size = ptr - command_buffer;
    const int rc = msg_->init_size (command_size);
    errno_assert (rc == 0);
    memcpy (msg_->data (), command_buffer, command_size);
    free (command_buffer);

283 284 285 286
    if (do_encryption)
        return encode_message (msg_);

    return 0;
287 288 289 290
}

int zmq::gssapi_mechanism_base_t::process_ready (msg_t *msg_)
{
Chris Laws's avatar
Chris Laws committed
291
    if (do_encryption) {
292 293 294 295
        const int rc = decode_message (msg_);
        if (rc != 0)
            return rc;
    }
296

297 298 299 300 301 302 303 304 305 306 307 308
    const unsigned char *ptr = static_cast <unsigned char *> (msg_->data ());
    size_t bytes_left = msg_->size ();

    if (bytes_left < 6 || memcmp (ptr, "\x05READY", 6)) {
        errno = EPROTO;
        return -1;
    }
    ptr += 6;
    bytes_left -= 6;
    return parse_metadata (ptr, bytes_left);
}

309 310 311 312 313
int zmq::gssapi_mechanism_base_t::acquire_credentials (char * service_name_, gss_cred_id_t * cred_)
{
    OM_uint32 maj_stat;
    OM_uint32 min_stat;
    gss_name_t server_name;
Chris Laws's avatar
Chris Laws committed
314

315 316 317
    gss_buffer_desc name_buf;
    name_buf.value = service_name_;
    name_buf.length = strlen ((char *) name_buf.value) + 1;
Chris Laws's avatar
Chris Laws committed
318

319
    maj_stat = gss_import_name (&min_stat, &name_buf,
320
                                GSS_C_NT_HOSTBASED_SERVICE, &server_name);
321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336

    if (maj_stat != GSS_S_COMPLETE)
        return -1;

    maj_stat = gss_acquire_cred (&min_stat, server_name, 0,
                                 GSS_C_NO_OID_SET, GSS_C_ACCEPT,
                                 cred_, NULL, NULL);

    if (maj_stat != GSS_S_COMPLETE)
        return -1;

    gss_release_name(&min_stat, &server_name);

    return 0;
}

Chris Laws's avatar
Chris Laws committed
337
#endif