2018-04-10 06:45:47 +12:00
/*
common routines for audit logging
Copyright ( C ) Andrew Bartlett < abartlet @ samba . org > 2018
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 3 of the License , or
( at your option ) any later version .
This program 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 General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
/*
* Error handling :
*
*/
# include "includes.h"
# include "librpc/ndr/libndr.h"
# include "lib/tsocket/tsocket.h"
# include "libcli/security/dom_sid.h"
# include "lib/messaging/messaging.h"
# include "auth/common_auth.h"
# include "audit_logging.h"
/*
* @ brief Get a human readable timestamp .
*
* Returns the current time formatted as
* " Tue, 14 Mar 2017 08:38:42.209028 NZDT "
*
* The returned string is allocated by talloc in the supplied context .
* It is the callers responsibility to free it .
*
* @ param mem_ctx talloc memory context that owns the returned string .
*
2018-07-13 09:14:09 +12:00
* @ return a human readable time stamp , or NULL in the event of an error .
2018-04-10 06:45:47 +12:00
*
*/
char * audit_get_timestamp ( TALLOC_CTX * frame )
{
char buffer [ 40 ] ; /* formatted time less usec and timezone */
char tz [ 10 ] ; /* formatted time zone */
struct tm * tm_info ; /* current local time */
struct timeval tv ; /* current system time */
2018-07-13 09:14:09 +12:00
int ret ; /* response code */
2018-04-10 06:45:47 +12:00
char * ts ; /* formatted time stamp */
2018-07-13 09:14:09 +12:00
ret = gettimeofday ( & tv , NULL ) ;
if ( ret ! = 0 ) {
2018-04-10 06:45:47 +12:00
DBG_ERR ( " Unable to get time of day: (%d) %s \n " ,
errno ,
strerror ( errno ) ) ;
return NULL ;
}
tm_info = localtime ( & tv . tv_sec ) ;
if ( tm_info = = NULL ) {
DBG_ERR ( " Unable to determine local time \n " ) ;
return NULL ;
}
strftime ( buffer , sizeof ( buffer ) - 1 , " %a, %d %b %Y %H:%M:%S " , tm_info ) ;
strftime ( tz , sizeof ( tz ) - 1 , " %Z " , tm_info ) ;
2021-09-08 12:57:03 +01:00
ts = talloc_asprintf ( frame , " %s.%06ld %s " , buffer , ( long ) tv . tv_usec , tz ) ;
2018-04-10 06:45:47 +12:00
if ( ts = = NULL ) {
DBG_ERR ( " Out of memory formatting time stamp \n " ) ;
}
return ts ;
}
2018-05-17 08:03:00 +12:00
/*
* @ brief write an audit message to the audit logs .
*
* Write a human readable text audit message to the samba logs .
*
* @ param prefix Text to be printed at the start of the log line
* @ param message The content of the log line .
* @ param debub_class The debug class to log the message with .
* @ param debug_level The debug level to log the message with .
*/
void audit_log_human_text ( const char * prefix ,
const char * message ,
int debug_class ,
int debug_level )
{
DEBUGC ( debug_class , debug_level , ( " %s %s \n " , prefix , message ) ) ;
}
2018-04-10 06:45:47 +12:00
2018-05-17 08:03:00 +12:00
# ifdef HAVE_JANSSON
2018-07-13 09:14:09 +12:00
/*
* Constant for empty json object initialisation
*/
const struct json_object json_empty_object = { . valid = false , . root = NULL } ;
2018-05-17 08:03:00 +12:00
/*
* @ brief write a json object to the samba audit logs .
*
* Write the json object to the audit logs as a formatted string
*
* @ param message The content of the log line .
* @ param debub_class The debug class to log the message with .
* @ param debug_level The debug level to log the message with .
*/
2018-12-13 13:53:08 +13:00
void audit_log_json ( struct json_object * message ,
2018-06-01 08:56:53 +12:00
int debug_class ,
int debug_level )
2018-05-17 08:03:00 +12:00
{
2018-12-14 15:40:20 +13:00
TALLOC_CTX * frame = NULL ;
2018-07-13 09:14:09 +12:00
char * s = NULL ;
if ( json_is_invalid ( message ) ) {
DBG_ERR ( " Invalid JSON object, unable to log \n " ) ;
return ;
}
2018-12-14 15:40:20 +13:00
frame = talloc_stackframe ( ) ;
s = json_to_string ( frame , message ) ;
2018-07-13 09:14:09 +12:00
if ( s = = NULL ) {
2018-12-13 13:53:08 +13:00
DBG_ERR ( " json_to_string returned NULL, "
" JSON audit message could not written \n " ) ;
2018-12-14 15:40:20 +13:00
TALLOC_FREE ( frame ) ;
2018-07-13 09:14:09 +12:00
return ;
}
2018-12-13 13:53:08 +13:00
/*
* This is very strange , but we call this routine to get a log
* output without the header . JSON logs all have timestamps
* so this only makes parsing harder .
*
* We push out the raw JSON blob without a prefix , consumers
* can find such lines by the leading {
*/
DEBUGADDC ( debug_class , debug_level , ( " %s \n " , s ) ) ;
2018-12-14 15:40:20 +13:00
TALLOC_FREE ( frame ) ;
2018-05-17 08:03:00 +12:00
}
2018-04-10 06:45:47 +12:00
/*
* @ brief get a connection to the messaging event server .
*
* Get a connection to the messaging event server registered by server_name .
*
* @ param msg_ctx a valid imessaging_context .
* @ param server_name name of messaging event server to connect to .
* @ param server_id The event server details to populate
*
* @ return NTSTATUS
*/
static NTSTATUS get_event_server (
struct imessaging_context * msg_ctx ,
const char * server_name ,
struct server_id * event_server )
{
NTSTATUS status ;
TALLOC_CTX * frame = talloc_stackframe ( ) ;
unsigned num_servers , i ;
struct server_id * servers ;
status = irpc_servers_byname (
msg_ctx ,
frame ,
server_name ,
& num_servers ,
& servers ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2019-10-11 08:28:30 +02:00
DBG_DEBUG ( " Failed to find the target '%s' on the message bus "
" to send JSON audit events to: %s \n " ,
server_name ,
nt_errstr ( status ) ) ;
2018-04-10 06:45:47 +12:00
TALLOC_FREE ( frame ) ;
return status ;
}
/*
* Select the first server that is listening , because we get
* connection refused as NT_STATUS_OBJECT_NAME_NOT_FOUND
* without waiting
*/
for ( i = 0 ; i < num_servers ; i + + ) {
status = imessaging_send (
msg_ctx ,
servers [ i ] ,
MSG_PING ,
& data_blob_null ) ;
if ( NT_STATUS_IS_OK ( status ) ) {
* event_server = servers [ i ] ;
TALLOC_FREE ( frame ) ;
return NT_STATUS_OK ;
}
}
DBG_NOTICE (
" Failed to find '%s' registered on the message bus to "
2018-06-25 14:48:27 +12:00
" send JSON audit events to: %s \n " ,
2018-04-10 06:45:47 +12:00
server_name ,
nt_errstr ( status ) ) ;
TALLOC_FREE ( frame ) ;
return NT_STATUS_OBJECT_NAME_NOT_FOUND ;
}
/*
* @ brief send an audit message to a messaging event server .
*
* Send the message to a registered and listening event server .
* Note : Any errors are logged , and the message is not sent . This is to ensure
* that a poorly behaved event server does not impact Samba .
*
* As it is possible to lose messages , especially during server
* shut down , currently this function is primarily intended for use
* in integration tests .
*
* @ param msg_ctx an imessaging_context , can be NULL in which case no message
* will be sent .
* @ param server_name the naname of the event server to send the message to .
* @ param messag_type A message type defined in librpc / idl / messaging . idl
* @ param message The message to send .
*
*/
void audit_message_send (
struct imessaging_context * msg_ctx ,
const char * server_name ,
uint32_t message_type ,
2018-05-17 08:03:00 +12:00
struct json_object * message )
2018-04-10 06:45:47 +12:00
{
2019-01-14 13:24:24 +01:00
struct server_id event_server = {
. pid = 0 ,
} ;
2018-04-10 06:45:47 +12:00
NTSTATUS status ;
2018-05-17 08:03:00 +12:00
const char * message_string = NULL ;
DATA_BLOB message_blob = data_blob_null ;
2018-07-13 09:14:09 +12:00
TALLOC_CTX * ctx = NULL ;
2018-04-10 06:45:47 +12:00
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( message ) ) {
DBG_ERR ( " Invalid JSON object, unable to send \n " ) ;
return ;
}
2018-04-10 06:45:47 +12:00
if ( msg_ctx = = NULL ) {
DBG_DEBUG ( " No messaging context \n " ) ;
return ;
}
2018-12-18 16:09:19 +01:00
ctx = talloc_new ( NULL ) ;
if ( ctx = = NULL ) {
DBG_ERR ( " Out of memory creating temporary context \n " ) ;
return ;
}
2018-04-10 06:45:47 +12:00
/* Need to refetch the address each time as the destination server may
* have disconnected and reconnected in the interim , in which case
* messages may get lost
*/
status = get_event_server ( msg_ctx , server_name , & event_server ) ;
2018-06-25 14:51:35 +12:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2018-05-17 08:03:00 +12:00
TALLOC_FREE ( ctx ) ;
2018-04-10 06:45:47 +12:00
return ;
}
2018-05-17 08:03:00 +12:00
message_string = json_to_string ( ctx , message ) ;
message_blob = data_blob_string_const ( message_string ) ;
2018-04-10 06:45:47 +12:00
status = imessaging_send (
msg_ctx ,
event_server ,
message_type ,
& message_blob ) ;
/*
* If the server crashed , try to find it again
*/
if ( NT_STATUS_EQUAL ( status , NT_STATUS_OBJECT_NAME_NOT_FOUND ) ) {
status = get_event_server ( msg_ctx , server_name , & event_server ) ;
2018-06-25 14:51:35 +12:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2018-05-17 08:03:00 +12:00
TALLOC_FREE ( ctx ) ;
2018-04-10 06:45:47 +12:00
return ;
}
imessaging_send (
msg_ctx ,
event_server ,
message_type ,
& message_blob ) ;
}
2018-05-17 08:03:00 +12:00
TALLOC_FREE ( ctx ) ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief Create a new struct json_object , wrapping a JSON Object .
*
* Create a new json object , the json_object wraps the underlying json
* implementations JSON Object representation .
*
2018-10-08 14:59:50 +02:00
* Free with a call to json_free_object , note that the jansson implementation
2018-04-10 06:45:47 +12:00
* allocates memory with malloc and not talloc .
*
2018-07-13 09:14:09 +12:00
* @ return a struct json_object , valid will be set to false if the object
2018-04-10 06:45:47 +12:00
* could not be created .
*
*/
struct json_object json_new_object ( void ) {
2018-07-13 09:14:09 +12:00
struct json_object object = json_empty_object ;
2018-04-10 06:45:47 +12:00
object . root = json_object ( ) ;
if ( object . root = = NULL ) {
2018-07-13 09:14:09 +12:00
object . valid = false ;
DBG_ERR ( " Unable to create JSON object \n " ) ;
return object ;
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
object . valid = true ;
2018-04-10 06:45:47 +12:00
return object ;
}
/*
* @ brief Create a new struct json_object wrapping a JSON Array .
*
* Create a new json object , the json_object wraps the underlying json
* implementations JSON Array representation .
*
2018-10-08 14:59:50 +02:00
* Free with a call to json_free_object , note that the jansson implementation
2018-04-10 06:45:47 +12:00
* allocates memory with malloc and not talloc .
*
* @ return a struct json_object , error will be set to true if the array
* could not be created .
*
*/
struct json_object json_new_array ( void ) {
2018-07-13 09:14:09 +12:00
struct json_object array = json_empty_object ;
2018-04-10 06:45:47 +12:00
array . root = json_array ( ) ;
if ( array . root = = NULL ) {
2018-07-13 09:14:09 +12:00
array . valid = false ;
DBG_ERR ( " Unable to create JSON array \n " ) ;
return array ;
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
array . valid = true ;
2018-04-10 06:45:47 +12:00
return array ;
}
/*
* @ brief free and invalidate a previously created JSON object .
*
* Release any resources owned by a json_object , and then mark the structure
* as invalid . It is safe to call this multiple times on an object .
*
*/
void json_free ( struct json_object * object )
{
if ( object - > root ! = NULL ) {
json_decref ( object - > root ) ;
}
object - > root = NULL ;
2018-07-13 09:14:09 +12:00
object - > valid = false ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief is the current JSON object invalid ?
*
* Check the state of the object to determine if it is invalid .
*
* @ return is the object valid ?
*
*/
2018-07-09 09:41:37 +02:00
bool json_is_invalid ( const struct json_object * object )
2018-04-10 06:45:47 +12:00
{
2018-07-13 09:14:09 +12:00
return ! object - > valid ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief Add an integer value to a JSON object .
*
* Add an integer value named ' name ' to the json object .
*
* @ param object the JSON object to be updated .
* @ param name the name of the value .
* @ param value the value .
*
2018-07-13 09:14:09 +12:00
* @ return 0 the operation was successful
* - 1 the operation failed
*
2018-04-10 06:45:47 +12:00
*/
2023-03-09 11:11:28 +08:00
int json_add_int ( struct json_object * object , const char * name , const json_int_t value )
2018-04-10 06:45:47 +12:00
{
2018-07-13 09:14:09 +12:00
int ret = 0 ;
json_t * integer = NULL ;
2018-04-10 06:45:47 +12:00
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( object ) ) {
2023-03-09 11:11:28 +08:00
DBG_ERR ( " Unable to add int [%s] value [%jd], "
2018-07-13 09:14:09 +12:00
" target object is invalid \n " ,
name ,
2023-03-09 11:11:28 +08:00
( intmax_t ) value ) ;
2018-07-13 09:14:09 +12:00
return JSON_ERROR ;
}
integer = json_integer ( value ) ;
if ( integer = = NULL ) {
2023-03-09 11:11:28 +08:00
DBG_ERR ( " Unable to create integer value [%s] value [%jd] \n " ,
2018-07-13 09:14:09 +12:00
name ,
2023-03-09 11:11:28 +08:00
( intmax_t ) value ) ;
2018-07-13 09:14:09 +12:00
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
ret = json_object_set_new ( object - > root , name , integer ) ;
if ( ret ! = 0 ) {
json_decref ( integer ) ;
2023-03-09 11:11:28 +08:00
DBG_ERR ( " Unable to add int [%s] value [%jd] \n " ,
name ,
( intmax_t ) value ) ;
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
return ret ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief Add a boolean value to a JSON object .
*
* Add a boolean value named ' name ' to the json object .
*
* @ param object the JSON object to be updated .
* @ param name the name .
* @ param value the value .
*
2018-07-13 09:14:09 +12:00
* @ return 0 the operation was successful
* - 1 the operation failed
*
2018-04-10 06:45:47 +12:00
*/
2018-07-13 09:14:09 +12:00
int json_add_bool ( struct json_object * object ,
const char * name ,
const bool value )
2018-04-10 06:45:47 +12:00
{
2018-07-13 09:14:09 +12:00
int ret = 0 ;
2018-04-10 06:45:47 +12:00
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Unable to add boolean [%s] value [%d], "
" target object is invalid \n " ,
name ,
value ) ;
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
ret = json_object_set_new ( object - > root , name , json_boolean ( value ) ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to add boolean [%s] value [%d] \n " , name , value ) ;
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
return ret ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief Add a string value to a JSON object .
*
* Add a string value named ' name ' to the json object .
*
* @ param object the JSON object to be updated .
* @ param name the name .
* @ param value the value .
*
2018-07-13 09:14:09 +12:00
* @ return 0 the operation was successful
* - 1 the operation failed
*
2018-04-10 06:45:47 +12:00
*/
2018-07-13 09:14:09 +12:00
int json_add_string ( struct json_object * object ,
const char * name ,
const char * value )
2018-04-10 06:45:47 +12:00
{
2018-07-13 09:14:09 +12:00
int ret = 0 ;
2018-04-10 06:45:47 +12:00
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Unable to add string [%s], target object is invalid \n " ,
name ) ;
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
if ( value ) {
2018-07-13 09:14:09 +12:00
json_t * string = json_string ( value ) ;
if ( string = = NULL ) {
DBG_ERR ( " Unable to add string [%s], "
" could not create string object \n " ,
name ) ;
return JSON_ERROR ;
}
ret = json_object_set_new ( object - > root , name , string ) ;
if ( ret ! = 0 ) {
json_decref ( string ) ;
DBG_ERR ( " Unable to add string [%s] \n " , name ) ;
return ret ;
}
2018-04-10 06:45:47 +12:00
} else {
2018-07-13 09:14:09 +12:00
ret = json_object_set_new ( object - > root , name , json_null ( ) ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to add null string [%s] \n " , name ) ;
return ret ;
}
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
return ret ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief Assert that the current JSON object is an array .
*
* Check that the current object is a JSON array , and if not
* invalidate the object . We also log an error message as this indicates
* bug in the calling code .
*
* @ param object the JSON object to be validated .
*/
void json_assert_is_array ( struct json_object * array ) {
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( array ) ) {
2018-04-10 06:45:47 +12:00
return ;
}
if ( json_is_array ( array - > root ) = = false ) {
DBG_ERR ( " JSON object is not an array \n " ) ;
2018-07-13 09:14:09 +12:00
array - > valid = false ;
2018-04-10 06:45:47 +12:00
return ;
}
}
/*
* @ brief Add a JSON object to a JSON object .
*
* Add a JSON object named ' name ' to the json object .
*
* @ param object the JSON object to be updated .
* @ param name the name .
* @ param value the value .
*
2018-07-13 09:14:09 +12:00
* @ return 0 the operation was successful
* - 1 the operation failed
*
2018-04-10 06:45:47 +12:00
*/
2018-07-13 09:14:09 +12:00
int json_add_object ( struct json_object * object ,
const char * name ,
struct json_object * value )
2018-04-10 06:45:47 +12:00
{
2018-07-13 09:14:09 +12:00
int ret = 0 ;
2018-04-10 06:45:47 +12:00
json_t * jv = NULL ;
2018-07-13 09:14:09 +12:00
if ( value ! = NULL & & json_is_invalid ( value ) ) {
DBG_ERR ( " Invalid JSON object [%s] supplied \n " , name ) ;
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Unable to add object [%s], target object is invalid \n " ,
name ) ;
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
jv = value = = NULL ? json_null ( ) : value - > root ;
if ( json_is_array ( object - > root ) ) {
2018-07-13 09:14:09 +12:00
ret = json_array_append_new ( object - > root , jv ) ;
2018-04-10 06:45:47 +12:00
} else if ( json_is_object ( object - > root ) ) {
2018-07-13 09:14:09 +12:00
ret = json_object_set_new ( object - > root , name , jv ) ;
2018-04-10 06:45:47 +12:00
} else {
DBG_ERR ( " Invalid JSON object type \n " ) ;
2018-07-13 09:14:09 +12:00
ret = JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
if ( ret ! = 0 ) {
2018-04-10 06:45:47 +12:00
DBG_ERR ( " Unable to add object [%s] \n " , name ) ;
}
2018-07-13 09:14:09 +12:00
return ret ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief Add a string to a JSON object , truncating if necessary .
*
*
* Add a string value named ' name ' to the json object , the string will be
* truncated if it is more than len characters long . If len is 0 the value
* is encoded as a JSON null .
*
*
* @ param object the JSON object to be updated .
* @ param name the name .
* @ param value the value .
* @ param len the maximum number of characters to be copied .
*
2018-07-13 09:14:09 +12:00
* @ return 0 the operation was successful
* - 1 the operation failed
*
2018-04-10 06:45:47 +12:00
*/
2018-07-13 09:14:09 +12:00
int json_add_stringn ( struct json_object * object ,
const char * name ,
const char * value ,
const size_t len )
2018-04-10 06:45:47 +12:00
{
2018-07-13 09:14:09 +12:00
int ret = 0 ;
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Unable to add string [%s], target object is invalid \n " ,
name ) ;
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
if ( value ! = NULL & & len > 0 ) {
2018-07-13 09:14:09 +12:00
json_t * string = NULL ;
2018-04-10 06:45:47 +12:00
char buffer [ len + 1 ] ;
2018-07-13 09:14:09 +12:00
2018-04-10 06:45:47 +12:00
strncpy ( buffer , value , len ) ;
buffer [ len ] = ' \0 ' ;
2018-07-13 09:14:09 +12:00
string = json_string ( buffer ) ;
if ( string = = NULL ) {
DBG_ERR ( " Unable to add string [%s], "
" could not create string object \n " ,
name ) ;
return JSON_ERROR ;
}
ret = json_object_set_new ( object - > root , name , string ) ;
if ( ret ! = 0 ) {
json_decref ( string ) ;
DBG_ERR ( " Unable to add string [%s] \n " , name ) ;
return ret ;
}
2018-04-10 06:45:47 +12:00
} else {
2018-07-13 09:14:09 +12:00
ret = json_object_set_new ( object - > root , name , json_null ( ) ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to add null string [%s] \n " , name ) ;
return ret ;
}
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
return ret ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief Add a version object to a JSON object
*
* Add a version object to the JSON object
* " version " : { " major " : 1 , " minor " : 0 }
*
* The version tag is intended to aid the processing of the JSON messages
* The major version number should change when an attribute is :
* - renamed
* - removed
* - its meaning changes
* - its contents change format
* The minor version should change whenever a new attribute is added and for
* minor bug fixes to an attributes content .
*
*
* @ param object the JSON object to be updated .
* @ param major the major version number
* @ param minor the minor version number
2018-07-13 09:14:09 +12:00
*
* @ return 0 the operation was successful
* - 1 the operation failed
2018-04-10 06:45:47 +12:00
*/
2018-07-13 09:14:09 +12:00
int json_add_version ( struct json_object * object , int major , int minor )
2018-04-10 06:45:47 +12:00
{
2018-07-13 09:14:09 +12:00
int ret = 0 ;
struct json_object version ;
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Unable to add version, target object is invalid \n " ) ;
return JSON_ERROR ;
}
version = json_new_object ( ) ;
if ( json_is_invalid ( & version ) ) {
DBG_ERR ( " Unable to add version, failed to create object \n " ) ;
return JSON_ERROR ;
}
ret = json_add_int ( & version , " major " , major ) ;
if ( ret ! = 0 ) {
json_free ( & version ) ;
return ret ;
}
ret = json_add_int ( & version , " minor " , minor ) ;
if ( ret ! = 0 ) {
json_free ( & version ) ;
return ret ;
}
ret = json_add_object ( object , " version " , & version ) ;
if ( ret ! = 0 ) {
json_free ( & version ) ;
return ret ;
}
return ret ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief add an ISO 8601 timestamp to the object .
*
* Add the current date and time as a timestamp in ISO 8601 format
* to a JSON object
*
* " timestamp " : " 2017-03-06T17:18:04.455081+1300 "
*
*
* @ param object the JSON object to be updated .
2018-07-13 09:14:09 +12:00
*
* @ return 0 the operation was successful
* - 1 the operation failed
2018-04-10 06:45:47 +12:00
*/
2018-07-13 09:14:09 +12:00
int json_add_timestamp ( struct json_object * object )
2018-04-10 06:45:47 +12:00
{
char buffer [ 40 ] ; /* formatted time less usec and timezone */
2018-05-16 13:59:55 +02:00
char timestamp [ 65 ] ; /* the formatted ISO 8601 time stamp */
2018-04-10 06:45:47 +12:00
char tz [ 10 ] ; /* formatted time zone */
struct tm * tm_info ; /* current local time */
struct timeval tv ; /* current system time */
int r ; /* response code from gettimeofday */
2018-07-13 09:14:09 +12:00
int ret ; /* return code from json operations */
2018-04-10 06:45:47 +12:00
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Unable to add time stamp, target object is invalid \n " ) ;
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
r = gettimeofday ( & tv , NULL ) ;
if ( r ) {
DBG_ERR ( " Unable to get time of day: (%d) %s \n " ,
errno ,
strerror ( errno ) ) ;
2018-07-13 09:14:09 +12:00
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
tm_info = localtime ( & tv . tv_sec ) ;
if ( tm_info = = NULL ) {
DBG_ERR ( " Unable to determine local time \n " ) ;
2018-07-13 09:14:09 +12:00
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
strftime ( buffer , sizeof ( buffer ) - 1 , " %Y-%m-%dT%T " , tm_info ) ;
strftime ( tz , sizeof ( tz ) - 1 , " %z " , tm_info ) ;
snprintf (
timestamp ,
sizeof ( timestamp ) ,
" %s.%06ld%s " ,
buffer ,
tv . tv_usec ,
tz ) ;
2018-07-13 09:14:09 +12:00
ret = json_add_string ( object , " timestamp " , timestamp ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to add time stamp to JSON object \n " ) ;
}
return ret ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief Add a tsocket_address to a JSON object
*
* Add the string representation of a Samba tsocket_address to the object .
*
* " localAddress " : " ipv6::::0 "
*
*
* @ param object the JSON object to be updated .
* @ param name the name .
* @ param address the tsocket_address .
*
2018-07-13 09:14:09 +12:00
* @ return 0 the operation was successful
* - 1 the operation failed
*
2018-04-10 06:45:47 +12:00
*/
2018-07-13 09:14:09 +12:00
int json_add_address ( struct json_object * object ,
const char * name ,
const struct tsocket_address * address )
2018-04-10 06:45:47 +12:00
{
2018-07-13 09:14:09 +12:00
int ret = 0 ;
2018-04-10 06:45:47 +12:00
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Unable to add address [%s], "
" target object is invalid \n " ,
name ) ;
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
2018-04-10 06:45:47 +12:00
if ( address = = NULL ) {
2018-07-13 09:14:09 +12:00
ret = json_object_set_new ( object - > root , name , json_null ( ) ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to add null address [%s] \n " , name ) ;
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
} else {
TALLOC_CTX * ctx = talloc_new ( NULL ) ;
char * s = NULL ;
2018-07-13 09:14:09 +12:00
if ( ctx = = NULL ) {
DBG_ERR ( " Out of memory adding address [%s] \n " , name ) ;
return JSON_ERROR ;
}
2018-04-10 06:45:47 +12:00
s = tsocket_address_string ( address , ctx ) ;
2018-07-13 09:14:09 +12:00
if ( s = = NULL ) {
DBG_ERR ( " Out of memory adding address [%s] \n " , name ) ;
TALLOC_FREE ( ctx ) ;
return JSON_ERROR ;
}
ret = json_add_string ( object , name , s ) ;
if ( ret ! = 0 ) {
DBG_ERR (
" Unable to add address [%s] value [%s] \n " , name , s ) ;
TALLOC_FREE ( ctx ) ;
return JSON_ERROR ;
}
2018-04-10 06:45:47 +12:00
TALLOC_FREE ( ctx ) ;
}
2018-07-13 09:14:09 +12:00
return ret ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief Add a formatted string representation of a sid to a json object .
*
* Add the string representation of a Samba sid to the object .
*
* " sid " : " S-1-5-18 "
*
*
* @ param object the JSON object to be updated .
* @ param name the name .
* @ param sid the sid
*
2018-07-13 09:14:09 +12:00
* @ return 0 the operation was successful
* - 1 the operation failed
*
2018-04-10 06:45:47 +12:00
*/
2018-07-13 09:14:09 +12:00
int json_add_sid ( struct json_object * object ,
const char * name ,
const struct dom_sid * sid )
2018-04-10 06:45:47 +12:00
{
2018-07-13 09:14:09 +12:00
int ret = 0 ;
2018-04-10 06:45:47 +12:00
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Unable to add SID [%s], "
" target object is invalid \n " ,
name ) ;
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
2018-04-10 06:45:47 +12:00
if ( sid = = NULL ) {
2018-07-13 09:14:09 +12:00
ret = json_object_set_new ( object - > root , name , json_null ( ) ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to add null SID [%s] \n " , name ) ;
return ret ;
2018-04-10 06:45:47 +12:00
}
} else {
2018-10-26 08:25:14 +02:00
struct dom_sid_buf sid_buf ;
2018-04-10 06:45:47 +12:00
2018-10-26 08:25:14 +02:00
ret = json_add_string (
object , name , dom_sid_str_buf ( sid , & sid_buf ) ) ;
2018-07-13 09:14:09 +12:00
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to add SID [%s] value [%s] \n " ,
name ,
2018-10-26 08:25:14 +02:00
sid_buf . buf ) ;
2018-07-13 09:14:09 +12:00
return ret ;
}
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
return ret ;
2018-04-10 06:45:47 +12:00
}
/*
* @ brief Add a formatted string representation of a guid to a json object .
*
* Add the string representation of a Samba GUID to the object .
*
* " guid " : " 1fb9f2ee-2a4d-4bf8-af8b-cb9d4529a9ab "
*
*
* @ param object the JSON object to be updated .
* @ param name the name .
* @ param guid the guid .
*
2018-07-13 09:14:09 +12:00
* @ return 0 the operation was successful
* - 1 the operation failed
*
2018-04-10 06:45:47 +12:00
*
*/
2018-07-13 09:14:09 +12:00
int json_add_guid ( struct json_object * object ,
const char * name ,
const struct GUID * guid )
2018-04-10 06:45:47 +12:00
{
2018-07-13 09:14:09 +12:00
int ret = 0 ;
2018-04-10 06:45:47 +12:00
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Unable to add GUID [%s], "
" target object is invalid \n " ,
name ) ;
return JSON_ERROR ;
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
2018-04-10 06:45:47 +12:00
if ( guid = = NULL ) {
2018-07-13 09:14:09 +12:00
ret = json_object_set_new ( object - > root , name , json_null ( ) ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to add null GUID [%s] \n " , name ) ;
return ret ;
2018-04-10 06:45:47 +12:00
}
} else {
char * guid_str ;
struct GUID_txt_buf guid_buff ;
guid_str = GUID_buf_string ( guid , & guid_buff ) ;
2018-07-13 09:14:09 +12:00
ret = json_add_string ( object , name , guid_str ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to guid GUID [%s] value [%s] \n " ,
name ,
guid_str ) ;
return ret ;
}
2018-04-10 06:45:47 +12:00
}
2018-07-13 09:14:09 +12:00
return ret ;
2018-04-10 06:45:47 +12:00
}
2022-03-22 16:06:37 +01:00
/*
* @ brief Replaces the object for a given key with a given json object .
*
* If key already exists , the value will be replaced . Otherwise the given
* value will be added under the given key .
*
* @ param object the JSON object to be updated .
* @ param key the key which will be updated .
* @ param new_obj the new value object to be inserted .
*
* @ return 0 the operation was successful
2023-03-31 11:03:08 +02:00
* - 1 the operation failed ( e . j . if one of the parameters is invalid )
2022-03-22 16:06:37 +01:00
*/
int json_update_object ( struct json_object * object ,
const char * key ,
struct json_object * new_obj )
{
int ret = 0 ;
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Unable to update key [%s], "
" target object is invalid \n " ,
key ) ;
return JSON_ERROR ;
}
if ( json_is_invalid ( new_obj ) ) {
DBG_ERR ( " Unable to update key [%s], "
" new object is invalid \n " ,
key ) ;
return JSON_ERROR ;
}
if ( key = = NULL ) {
DBG_ERR ( " Unable to add null String as key \n " ) ;
return JSON_ERROR ;
}
ret = json_object_set ( object - > root , key , new_obj - > root ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to update object \n " ) ;
return ret ;
}
return ret ;
}
2018-04-10 06:45:47 +12:00
/*
* @ brief Convert a JSON object into a string
*
2021-05-07 11:13:51 +02:00
* Convert the json object into a string suitable for printing on a log line ,
2018-04-10 06:45:47 +12:00
* i . e . with no embedded line breaks .
*
2018-07-13 09:14:09 +12:00
* If the object is invalid it logs an error and returns NULL .
2018-04-10 06:45:47 +12:00
*
* @ param mem_ctx the talloc memory context owning the returned string
* @ param object the json object .
*
* @ return A string representation of the object or NULL if the object
* is invalid .
*/
2018-07-09 09:41:37 +02:00
char * json_to_string ( TALLOC_CTX * mem_ctx , const struct json_object * object )
2018-04-10 06:45:47 +12:00
{
char * json = NULL ;
char * json_string = NULL ;
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Invalid JSON object, unable to convert to string \n " ) ;
return NULL ;
}
if ( object - > root = = NULL ) {
2018-04-10 06:45:47 +12:00
return NULL ;
}
/*
* json_dumps uses malloc , so need to call free ( json ) to release
* the memory
*/
json = json_dumps ( object - > root , 0 ) ;
if ( json = = NULL ) {
DBG_ERR ( " Unable to convert JSON object to string \n " ) ;
return NULL ;
}
json_string = talloc_strdup ( mem_ctx , json ) ;
if ( json_string = = NULL ) {
free ( json ) ;
DBG_ERR ( " Unable to copy JSON object string to talloc string \n " ) ;
return NULL ;
}
free ( json ) ;
return json_string ;
}
2018-05-17 08:03:00 +12:00
/*
* @ brief get a json array named " name " from the json object .
*
* Get the array attribute named name , creating it if it does not exist .
*
* @ param object the json object .
* @ param name the name of the array attribute
*
* @ return The array object , will be created if it did not exist .
*/
2018-07-13 09:14:09 +12:00
struct json_object json_get_array ( struct json_object * object , const char * name )
2018-05-17 08:03:00 +12:00
{
2018-07-13 09:14:09 +12:00
struct json_object array = json_empty_object ;
2018-05-17 08:03:00 +12:00
json_t * a = NULL ;
2018-07-13 09:14:09 +12:00
int ret = 0 ;
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Invalid JSON object, unable to get array [%s] \n " ,
name ) ;
json_free ( & array ) ;
return array ;
}
2018-05-17 08:03:00 +12:00
2018-07-13 09:14:09 +12:00
array = json_new_array ( ) ;
if ( json_is_invalid ( & array ) ) {
DBG_ERR ( " Unable to create new array for [%s] \n " , name ) ;
2018-05-17 08:03:00 +12:00
return array ;
}
a = json_object_get ( object - > root , name ) ;
if ( a = = NULL ) {
return array ;
}
2018-07-13 09:14:09 +12:00
ret = json_array_extend ( array . root , a ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to get array [%s] \n " , name ) ;
json_free ( & array ) ;
return array ;
}
2018-05-17 08:03:00 +12:00
return array ;
}
/*
* @ brief get a json object named " name " from the json object .
*
* Get the object attribute named name , creating it if it does not exist .
*
* @ param object the json object .
* @ param name the name of the object attribute
*
* @ return The object , will be created if it did not exist .
*/
2018-07-13 09:14:09 +12:00
struct json_object json_get_object ( struct json_object * object , const char * name )
2018-05-17 08:03:00 +12:00
{
struct json_object o = json_new_object ( ) ;
json_t * v = NULL ;
2018-07-13 09:14:09 +12:00
int ret = 0 ;
2018-05-17 08:03:00 +12:00
2018-07-13 09:14:09 +12:00
if ( json_is_invalid ( object ) ) {
DBG_ERR ( " Invalid JSON object, unable to get object [%s] \n " ,
name ) ;
json_free ( & o ) ;
2018-05-17 08:03:00 +12:00
return o ;
}
v = json_object_get ( object - > root , name ) ;
if ( v = = NULL ) {
return o ;
}
2018-07-13 09:14:09 +12:00
ret = json_object_update ( o . root , v ) ;
if ( ret ! = 0 ) {
DBG_ERR ( " Unable to get object [%s] \n " , name ) ;
json_free ( & o ) ;
return o ;
}
2018-05-17 08:03:00 +12:00
return o ;
}
2018-04-10 06:45:47 +12:00
# endif