/*
 *
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 *
 */

#include "proton/delivery.hpp"

#include "proton/receiver.hpp"

#include "proton_bits.hpp"

#include <proton/delivery.h>
#include <proton/disposition.h>
#include <proton/link.h>

#include "types_internal.hpp"

#include "proton/binary.hpp"

#include <proton/session.hpp>

namespace {

void settle_delivery(pn_delivery_t* o, uint64_t state) {
    proton::session session = proton::make_wrapper(o).session();
    if(session.transaction_is_declared()) {
        // Transactional disposition
        auto disp = pn_transactional_disposition(pn_delivery_local(o));
        pn_transactional_disposition_set_id(disp, pn_bytes(session.transaction_id()));
        pn_transactional_disposition_set_outcome_type(disp, state);
        pn_delivery_update(o, PN_TRANSACTIONAL_STATE);
        return;
    }
    pn_delivery_update(o, state);
    pn_delivery_settle(o);
}

}

namespace proton {

delivery::delivery(pn_delivery_t* d): transfer(make_wrapper(d)) {}
receiver delivery::receiver() const { return make_wrapper<class receiver>(pn_delivery_link(pn_object())); }
binary delivery::tag() const { return bin(pn_delivery_tag(pn_object())); }
delivery::~delivery() = default;
void delivery::accept() { settle_delivery(pn_object(), ACCEPTED); }
void delivery::reject() { settle_delivery(pn_object(), REJECTED); }
void delivery::release() { settle_delivery(pn_object(), RELEASED); }
void delivery::modify() { pn_disposition_set_failed(pn_delivery_local(pn_object()), true); settle_delivery(pn_object(), MODIFIED); }

delivery_iterator delivery_iterator::operator++() {
    if (!!obj_) {
        pn_delivery_t* next = pn_unsettled_next(unwrap(obj_));
        obj_ = make_wrapper<delivery>(next);
    }
    return *this;
}

}
