/* -*- Mode: C; tab-width: 4 -*-
 *
 * Copyright (c) 2009 Apple Computer, Inc. All rights reserved.
 *
 * Licensed 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 "stdafx.h"

#include "TXTRecord.h"

#include "StringServices.h"

#include <DebugServices.h>





// CTXTRecord





STDMETHODIMP CTXTRecord::SetValue(BSTR key, VARIANT value)

{

	std::string			keyUTF8;

	ByteArray			valueArray;

	BOOL				ok;

	DNSServiceErrorType	err;

	HRESULT				hr = S_OK;



	if ( !m_allocated )

	{

		TXTRecordCreate( &m_tref, 0, NULL );

		m_allocated = TRUE;

	}



	ok = BSTRToUTF8( key, keyUTF8 );

	require_action( ok, exit, hr = S_FALSE );



	ok = VariantToByteArray( &value, valueArray );

	require_action( ok, exit, hr = S_FALSE );



	err = TXTRecordSetValue( &m_tref, keyUTF8.c_str(), ( uint8_t ) valueArray.size(), &valueArray[ 0 ] );

	require_action( !err, exit, hr = S_FALSE );



exit:



	return hr;

}



STDMETHODIMP CTXTRecord::RemoveValue(BSTR key)

{

	HRESULT hr = S_OK;



	if ( m_allocated )

	{

		std::string			keyUTF8;

		BOOL				ok;

		DNSServiceErrorType	err;



		ok = BSTRToUTF8( key, keyUTF8 );

		require_action( ok, exit, hr = S_FALSE );



		err = TXTRecordRemoveValue( &m_tref, keyUTF8.c_str() );

		require_action( !err, exit, hr = S_FALSE );

	}



exit:



	return hr;

}



STDMETHODIMP CTXTRecord::ContainsKey(BSTR key, VARIANT_BOOL* retval)

{

	std::string keyUTF8;

	int			ret	= 0;

	HRESULT		err	= S_OK;



	if ( m_byteArray.size() > 0 )

	{

		BOOL ok;



		ok = BSTRToUTF8( key, keyUTF8 );

		require_action( ok, exit, err = S_FALSE );



		ret = TXTRecordContainsKey( ( uint16_t ) m_byteArray.size(), &m_byteArray[ 0 ], keyUTF8.c_str() );

	}



	*retval = ( ret ) ? VARIANT_TRUE : VARIANT_FALSE;



exit:



	return err;

}



STDMETHODIMP CTXTRecord::GetValueForKey(BSTR key, VARIANT* value)

{

	std::string		keyUTF8;

	const void	*	rawValue;

	uint8_t			rawValueLen;

	BOOL			ok	= TRUE;

	HRESULT			hr	= S_OK;



	VariantClear( value );



	if ( m_byteArray.size() > 0 )

	{

		ok = BSTRToUTF8( key, keyUTF8 );

		require_action( ok, exit, hr = S_FALSE );



		rawValue = TXTRecordGetValuePtr( ( uint16_t ) m_byteArray.size(), &m_byteArray[ 0 ], keyUTF8.c_str(), &rawValueLen );



		if ( rawValue )

		{

			ok = ByteArrayToVariant( rawValue, rawValueLen, value );

			require_action( ok, exit, hr = S_FALSE );

		}

	}



exit:



	return hr;

}



STDMETHODIMP CTXTRecord::GetCount(ULONG* count)

{

	*count = 0;



	if ( m_byteArray.size() > 0 )

	{

		*count = TXTRecordGetCount( ( uint16_t ) m_byteArray.size(), &m_byteArray[ 0 ] );

	}



	return S_OK;

}



STDMETHODIMP CTXTRecord::GetKeyAtIndex(ULONG index, BSTR* retval)

{

	char				keyBuf[ 64 ];

	uint8_t				rawValueLen;

	const void		*	rawValue;

	CComBSTR			temp;

	DNSServiceErrorType	err;

	BOOL				ok;

	HRESULT				hr = S_OK;



	err = TXTRecordGetItemAtIndex( ( uint16_t ) m_byteArray.size(), &m_byteArray[ 0 ], ( uint16_t ) index, sizeof( keyBuf ), keyBuf, &rawValueLen, &rawValue );

	require_action( !err, exit, hr = S_FALSE );



	ok = UTF8ToBSTR( keyBuf, temp );

	require_action( ok, exit, hr = S_FALSE );



	*retval = temp;



exit:



	return hr;

}



STDMETHODIMP CTXTRecord::GetValueAtIndex(ULONG index, VARIANT* retval)

{

	char				keyBuf[ 64 ];

	uint8_t				rawValueLen;

	const void		*	rawValue;

	CComBSTR			temp;

	DNSServiceErrorType	err;

	BOOL				ok;

	HRESULT				hr = S_OK;



	err = TXTRecordGetItemAtIndex( ( uint16_t ) m_byteArray.size(), &m_byteArray[ 0 ], ( uint16_t ) index, sizeof( keyBuf ), keyBuf, &rawValueLen, &rawValue );

	require_action( !err, exit, hr = S_FALSE );



	ok = ByteArrayToVariant( rawValue, rawValueLen, retval );

	require_action( ok, exit, hr = S_FALSE );



exit:



	return hr;

}





void

CTXTRecord::SetBytes

	(

	const unsigned char	*	bytes,

	uint16_t				len

	)

{

	check ( bytes != NULL );

	check( len );



	m_byteArray.reserve( len );

	m_byteArray.assign( bytes, bytes + len );

}