dssync.c

来自「samba最新软件」· C语言 代码 · 共 835 行 · 第 1/2 页

C
835
字号
static void test_analyse_objects(struct torture_context *tctx, 				 struct DsSyncTest *ctx,				 const DATA_BLOB *gensec_skey,				 struct drsuapi_DsReplicaObjectListItemEx *cur){	static uint32_t object_id;	const char *save_values_dir;	if (!lp_parm_bool(tctx->lp_ctx, NULL, "dssync", "print_pwd_blobs", false)) {		return;		}	save_values_dir = lp_parm_string(tctx->lp_ctx, NULL, "dssync", "save_pwd_blobs_dir");	for (; cur; cur = cur->next_object) {		const char *dn;		struct dom_sid *sid = NULL;		uint32_t rid = 0;		bool dn_printed = false;		uint32_t i;		if (!cur->object.identifier) continue;		dn = cur->object.identifier->dn;		if (cur->object.identifier->sid.num_auths > 0) {			sid = &cur->object.identifier->sid;			rid = sid->sub_auths[sid->num_auths - 1];		}		for (i=0; i < cur->object.attribute_ctr.num_attributes; i++) {			const char *name = NULL;			bool rcrypt = false;			DATA_BLOB *enc_data = NULL;			DATA_BLOB plain_data;			struct drsuapi_DsReplicaAttribute *attr;			attr = &cur->object.attribute_ctr.attributes[i];			switch (attr->attid) {			case DRSUAPI_ATTRIBUTE_dBCSPwd:				name	= "dBCSPwd";				rcrypt	= true;				break;			case DRSUAPI_ATTRIBUTE_unicodePwd:				name	= "unicodePwd";				rcrypt	= true;				break;			case DRSUAPI_ATTRIBUTE_ntPwdHistory:				name	= "ntPwdHistory";				rcrypt	= true;				break;			case DRSUAPI_ATTRIBUTE_lmPwdHistory:				name	= "lmPwdHistory";				rcrypt	= true;				break;			case DRSUAPI_ATTRIBUTE_supplementalCredentials:				name	= "supplementalCredentials";				break;			case DRSUAPI_ATTRIBUTE_priorValue:				name	= "priorValue";				break;			case DRSUAPI_ATTRIBUTE_currentValue:				name	= "currentValue";				break;			case DRSUAPI_ATTRIBUTE_trustAuthOutgoing:				name	= "trustAuthOutgoing";				break;			case DRSUAPI_ATTRIBUTE_trustAuthIncoming:				name	= "trustAuthIncoming";				break;			case DRSUAPI_ATTRIBUTE_initialAuthOutgoing:				name	= "initialAuthOutgoing";				break;			case DRSUAPI_ATTRIBUTE_initialAuthIncoming:				name	= "initialAuthIncoming";				break;			default:				continue;			}			if (attr->value_ctr.num_values != 1) continue;			if (!attr->value_ctr.values[0].blob) continue;			enc_data = attr->value_ctr.values[0].blob;			ZERO_STRUCT(plain_data);			plain_data = decrypt_blob(ctx, gensec_skey, rcrypt,						  cur->object.identifier, rid,						  enc_data);			if (!dn_printed) {				object_id++;				DEBUG(0,("DN[%u] %s\n", object_id, dn));				dn_printed = true;			}			DEBUGADD(0,("ATTR: %s enc.length=%lu plain.length=%lu\n",				    name, (long)enc_data->length, (long)plain_data.length));			if (plain_data.length) {				dump_data(0, plain_data.data, plain_data.length);				if (save_values_dir) {					char *fname;					fname = talloc_asprintf(ctx, "%s/%s%02d",								save_values_dir,								name, object_id);					if (fname) {						bool ok;						ok = file_save(fname, plain_data.data, plain_data.length);						if (!ok) {							DEBUGADD(0,("Failed to save '%s'\n", fname));						}					}					talloc_free(fname);				}			} else {				dump_data(0, enc_data->data, enc_data->length);			}		}	}}static bool test_FetchData(struct torture_context *tctx, struct DsSyncTest *ctx){	NTSTATUS status;	bool ret = true;	int i, y = 0;	uint64_t highest_usn = 0;	const char *partition = NULL;	struct drsuapi_DsGetNCChanges r;	struct drsuapi_DsReplicaObjectIdentifier nc;	struct drsuapi_DsGetNCChangesCtr1 *ctr1 = NULL;	struct drsuapi_DsGetNCChangesCtr6 *ctr6 = NULL;	int32_t out_level = 0;	struct GUID null_guid;	struct dom_sid null_sid;	DATA_BLOB gensec_skey;	struct {		int32_t level;	} array[] = {/*		{			5		},*/		{			8		}	};	ZERO_STRUCT(null_guid);	ZERO_STRUCT(null_sid);	partition = lp_parm_string(tctx->lp_ctx, NULL, "dssync", "partition");	if (partition == NULL) {		partition = ctx->domain_dn;		printf("dssync:partition not specified, defaulting to %s.\n", ctx->domain_dn);	}	highest_usn = lp_parm_int(tctx->lp_ctx, NULL, "dssync", "highest_usn", 0);	array[0].level = lp_parm_int(tctx->lp_ctx, NULL, "dssync", "get_nc_changes_level", array[0].level);	if (lp_parm_bool(tctx->lp_ctx, NULL, "dssync", "print_pwd_blobs", false)) {		const struct samr_Password *nthash;		nthash = cli_credentials_get_nt_hash(ctx->new_dc.credentials, ctx);		if (nthash) {			dump_data_pw("CREDENTIALS nthash:", nthash->hash, sizeof(nthash->hash));		}	}	status = gensec_session_key(ctx->new_dc.drsuapi.pipe->conn->security_state.generic_state,				    &gensec_skey);	if (!NT_STATUS_IS_OK(status)) {		printf("failed to get gensec session key: %s\n", nt_errstr(status));		return false;	}	for (i=0; i < ARRAY_SIZE(array); i++) {		printf("testing DsGetNCChanges level %d\n",			array[i].level);		r.in.bind_handle	= &ctx->new_dc.drsuapi.bind_handle;		r.in.level		= &array[i].level;		switch (*r.in.level) {		case 5:			nc.guid	= null_guid;			nc.sid	= null_sid;			nc.dn	= partition; 			r.in.req.req5.destination_dsa_guid		= ctx->new_dc.invocation_id;			r.in.req.req5.source_dsa_invocation_id		= null_guid;			r.in.req.req5.naming_context			= &nc;			r.in.req.req5.highwatermark.tmp_highest_usn	= highest_usn;			r.in.req.req5.highwatermark.reserved_usn	= 0;			r.in.req.req5.highwatermark.highest_usn		= highest_usn;			r.in.req.req5.uptodateness_vector		= NULL;			r.in.req.req5.replica_flags			= 0;			if (lp_parm_bool(tctx->lp_ctx, NULL, "dssync", "compression", false)) {				r.in.req.req5.replica_flags		|= DRSUAPI_DS_REPLICA_NEIGHBOUR_COMPRESS_CHANGES;			}			if (lp_parm_bool(tctx->lp_ctx, NULL, "dssync", "neighbour_writeable", true)) {				r.in.req.req5.replica_flags		|= DRSUAPI_DS_REPLICA_NEIGHBOUR_WRITEABLE;			}			r.in.req.req5.replica_flags			|= DRSUAPI_DS_REPLICA_NEIGHBOUR_SYNC_ON_STARTUP									| DRSUAPI_DS_REPLICA_NEIGHBOUR_DO_SCHEDULED_SYNCS									| DRSUAPI_DS_REPLICA_NEIGHBOUR_RETURN_OBJECT_PARENTS									| DRSUAPI_DS_REPLICA_NEIGHBOUR_NEVER_SYNCED									;			r.in.req.req5.max_object_count			= 133;			r.in.req.req5.max_ndr_size			= 1336770;			r.in.req.req5.unknown4				= 0;			r.in.req.req5.h1				= 0;			break;		case 8:			nc.guid	= null_guid;			nc.sid	= null_sid;			nc.dn	= partition; 			/* nc.dn can be set to any other ad partition */						r.in.req.req8.destination_dsa_guid		= ctx->new_dc.invocation_id;			r.in.req.req8.source_dsa_invocation_id		= null_guid;			r.in.req.req8.naming_context			= &nc;			r.in.req.req8.highwatermark.tmp_highest_usn	= highest_usn;			r.in.req.req8.highwatermark.reserved_usn	= 0;			r.in.req.req8.highwatermark.highest_usn		= highest_usn;			r.in.req.req8.uptodateness_vector		= NULL;			r.in.req.req8.replica_flags			= 0;			if (lp_parm_bool(tctx->lp_ctx, NULL, "dssync", "compression", false)) {				r.in.req.req8.replica_flags		|= DRSUAPI_DS_REPLICA_NEIGHBOUR_COMPRESS_CHANGES;			}			if (lp_parm_bool(tctx->lp_ctx, NULL, "dssync", "neighbour_writeable", true)) {				r.in.req.req8.replica_flags		|= DRSUAPI_DS_REPLICA_NEIGHBOUR_WRITEABLE;			}			r.in.req.req8.replica_flags			|= DRSUAPI_DS_REPLICA_NEIGHBOUR_SYNC_ON_STARTUP									| DRSUAPI_DS_REPLICA_NEIGHBOUR_DO_SCHEDULED_SYNCS									| DRSUAPI_DS_REPLICA_NEIGHBOUR_RETURN_OBJECT_PARENTS									| DRSUAPI_DS_REPLICA_NEIGHBOUR_NEVER_SYNCED									;			r.in.req.req8.max_object_count			= 402;			r.in.req.req8.max_ndr_size			= 402116;			r.in.req.req8.unknown4				= 0;			r.in.req.req8.h1				= 0;			r.in.req.req8.unique_ptr1			= 0;			r.in.req.req8.unique_ptr2			= 0;			r.in.req.req8.mapping_ctr.num_mappings		= 0;			r.in.req.req8.mapping_ctr.mappings		= NULL;			break;		}				printf("Dumping AD partition: %s\n", nc.dn);		for (y=0; ;y++) {			int32_t _level = 0;			ZERO_STRUCT(r.out);			r.out.level = &_level;			if (*r.in.level == 5) {				DEBUG(0,("start[%d] tmp_higest_usn: %llu , highest_usn: %llu\n",y,					(long long)r.in.req.req5.highwatermark.tmp_highest_usn,					(long long)r.in.req.req5.highwatermark.highest_usn));			}			if (*r.in.level == 8) {				DEBUG(0,("start[%d] tmp_higest_usn: %llu , highest_usn: %llu\n",y,					(long long)r.in.req.req8.highwatermark.tmp_highest_usn,					(long long)r.in.req.req8.highwatermark.highest_usn));			}			status = dcerpc_drsuapi_DsGetNCChanges(ctx->new_dc.drsuapi.pipe, ctx, &r);			if (!NT_STATUS_IS_OK(status)) {				const char *errstr = nt_errstr(status);				if (NT_STATUS_EQUAL(status, NT_STATUS_NET_WRITE_FAULT)) {					errstr = dcerpc_errstr(ctx, ctx->new_dc.drsuapi.pipe->last_fault_code);				}				printf("dcerpc_drsuapi_DsGetNCChanges failed - %s\n", errstr);				ret = false;			} else if (!W_ERROR_IS_OK(r.out.result)) {				printf("DsGetNCChanges failed - %s\n", win_errstr(r.out.result));				ret = false;			}			if (ret == true && *r.out.level == 1) {				out_level = 1;				ctr1 = &r.out.ctr.ctr1;			} else if (ret == true && *r.out.level == 2) {				out_level = 1;				ctr1 = r.out.ctr.ctr2.ctr.mszip1.ctr1;			}			if (out_level == 1) {				DEBUG(0,("end[%d] tmp_highest_usn: %llu , highest_usn: %llu\n",y,					(long long)ctr1->new_highwatermark.tmp_highest_usn,					(long long)ctr1->new_highwatermark.highest_usn));				test_analyse_objects(tctx, ctx, &gensec_skey, ctr1->first_object);				if (ctr1->new_highwatermark.tmp_highest_usn > ctr1->new_highwatermark.highest_usn) {					r.in.req.req5.highwatermark = ctr1->new_highwatermark;					continue;				}			}			if (ret == true && *r.out.level == 6) {				out_level = 6;				ctr6 = &r.out.ctr.ctr6;			} else if (ret == true && *r.out.level == 7				   && r.out.ctr.ctr7.level == 6				   && r.out.ctr.ctr7.type == DRSUAPI_COMPRESSION_TYPE_MSZIP) {				out_level = 6;				ctr6 = r.out.ctr.ctr7.ctr.mszip6.ctr6;			}			if (out_level == 6) {				DEBUG(0,("end[%d] tmp_highest_usn: %llu , highest_usn: %llu\n",y,					(long long)ctr6->new_highwatermark.tmp_highest_usn,					(long long)ctr6->new_highwatermark.highest_usn));				test_analyse_objects(tctx, ctx, &gensec_skey, ctr6->first_object);				if (ctr6->new_highwatermark.tmp_highest_usn > ctr6->new_highwatermark.highest_usn) {					r.in.req.req8.highwatermark = ctr6->new_highwatermark;					continue;				}			}			break;		}	}	return ret;}static bool test_FetchNT4Data(struct torture_context *tctx, 			      struct DsSyncTest *ctx){	NTSTATUS status;	bool ret = true;	struct drsuapi_DsGetNT4ChangeLog r;	struct GUID null_guid;	struct dom_sid null_sid;	DATA_BLOB cookie;	ZERO_STRUCT(null_guid);	ZERO_STRUCT(null_sid);	ZERO_STRUCT(cookie);	ZERO_STRUCT(r);	r.in.bind_handle	= &ctx->new_dc.drsuapi.bind_handle;	r.in.level		= 1;	r.in.req.req1.unknown1	= lp_parm_int(tctx->lp_ctx, NULL, "dssync", "nt4-1", 3);	r.in.req.req1.unknown2	= lp_parm_int(tctx->lp_ctx, NULL, "dssync", "nt4-2", 0x00004000);	while (1) {		r.in.req.req1.length	= cookie.length;		r.in.req.req1.data	= cookie.data;		status = dcerpc_drsuapi_DsGetNT4ChangeLog(ctx->new_dc.drsuapi.pipe, ctx, &r);		if (!NT_STATUS_IS_OK(status)) {			const char *errstr = nt_errstr(status);			if (NT_STATUS_EQUAL(status, NT_STATUS_NET_WRITE_FAULT)) {				errstr = dcerpc_errstr(ctx, ctx->new_dc.drsuapi.pipe->last_fault_code);			}			printf("dcerpc_drsuapi_DsGetNT4ChangeLog failed - %s\n", errstr);			ret = false;		} else if (W_ERROR_EQUAL(r.out.result, WERR_INVALID_DOMAIN_ROLE)) {			printf("DsGetNT4ChangeLog not supported by target server\n");			break;		} else if (!W_ERROR_IS_OK(r.out.result)) {			printf("DsGetNT4ChangeLog failed - %s\n", win_errstr(r.out.result));			ret = false;		} else if (r.out.level != 1) {			printf("DsGetNT4ChangeLog unknown level - %u\n", r.out.level);			ret = false;		} else if (NT_STATUS_IS_OK(r.out.info.info1.status)) {		} else if (NT_STATUS_EQUAL(r.out.info.info1.status, STATUS_MORE_ENTRIES)) {			cookie.length	= r.out.info.info1.length1;			cookie.data	= r.out.info.info1.data1;			continue;		} else {			printf("DsGetNT4ChangeLog failed - %s\n", nt_errstr(r.out.info.info1.status));			ret = false;		}		break;	}	return ret;}bool torture_rpc_dssync(struct torture_context *torture){	bool ret = true;	TALLOC_CTX *mem_ctx;	struct DsSyncTest *ctx;		mem_ctx = talloc_init("torture_rpc_dssync");	ctx = test_create_context(torture);		ret &= _test_DsBind(torture, ctx, ctx->admin.credentials, &ctx->admin.drsuapi);	if (!ret) {		return ret;	}	ret &= test_LDAPBind(torture, ctx, ctx->admin.credentials, &ctx->admin.ldap);	if (!ret) {		return ret;	}	ret &= test_GetInfo(torture, ctx);	ret &= _test_DsBind(torture, ctx, ctx->new_dc.credentials, &ctx->new_dc.drsuapi);	if (!ret) {		return ret;	}	ret &= test_FetchData(torture, ctx);	ret &= test_FetchNT4Data(torture, ctx);	return ret;}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?