ssb: Don't schedule duplicate history stream requests for the same account.  Changes how we schedule idle work.  Let's see if this is better.
	
		
			
	
		
	
	
		
	
		
			All checks were successful
		
		
	
	
		
			
				
	
				Build Tilde Friends / Build-All (push) Successful in 22m35s
				
			
		
		
	
	
				
					
				
			
		
			All checks were successful
		
		
	
	Build Tilde Friends / Build-All (push) Successful in 22m35s
				
			This commit is contained in:
		
							
								
								
									
										42
									
								
								src/ssb.c
									
									
									
									
									
								
							
							
						
						
									
										42
									
								
								src/ssb.c
									
									
									
									
									
								
							| @@ -261,6 +261,7 @@ typedef struct _tf_ssb_connection_message_request_t | ||||
|  | ||||
| typedef struct _tf_ssb_connection_scheduled_t | ||||
| { | ||||
| 	char key[k_id_base64_len]; | ||||
| 	tf_ssb_scheduled_callback_t* callback; | ||||
| 	void* user_data; | ||||
| } tf_ssb_connection_scheduled_t; | ||||
| @@ -595,24 +596,55 @@ static void _tf_ssb_connection_dispatch_scheduled(tf_ssb_connection_t* connectio | ||||
| { | ||||
| 	while (((connection->active_write_count == 0 && connection->read_back_pressure == 0) || connection->closing) && connection->scheduled_count && connection->scheduled) | ||||
| 	{ | ||||
| 		tf_ssb_connection_scheduled_t scheduled = connection->scheduled[0]; | ||||
| 		memmove(connection->scheduled, connection->scheduled + 1, sizeof(tf_ssb_connection_scheduled_t) * (connection->scheduled_count - 1)); | ||||
| 		int index = uv_hrtime() % connection->scheduled_count; | ||||
| 		tf_ssb_connection_scheduled_t scheduled = connection->scheduled[index]; | ||||
| 		if (index != connection->scheduled_count - 1) | ||||
| 		{ | ||||
| 			memmove(connection->scheduled + index, connection->scheduled + index + 1, sizeof(tf_ssb_connection_scheduled_t) * (connection->scheduled_count - index - 1)); | ||||
| 		} | ||||
| 		connection->scheduled_count--; | ||||
| 		tf_trace_begin(connection->ssb->trace, "scheduled callback"); | ||||
| 		PRE_CALLBACK(connection->ssb, scheduled.callback); | ||||
| 		scheduled.callback(connection, scheduled.user_data); | ||||
| 		scheduled.callback(connection, false, scheduled.user_data); | ||||
| 		POST_CALLBACK(connection->ssb, scheduled.callback); | ||||
| 		tf_trace_end(connection->ssb->trace); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| void tf_ssb_connection_schedule_idle(tf_ssb_connection_t* connection, tf_ssb_scheduled_callback_t* callback, void* user_data) | ||||
| static int _tf_ssb_connection_scheduled_compare(const void* a, const void* b) | ||||
| { | ||||
| 	const char* key = a; | ||||
| 	const tf_ssb_connection_scheduled_t* scheduled = b; | ||||
| 	return strcmp(key, scheduled->key); | ||||
| } | ||||
|  | ||||
| void tf_ssb_connection_schedule_idle(tf_ssb_connection_t* connection, const char* key, tf_ssb_scheduled_callback_t* callback, void* user_data) | ||||
| { | ||||
| 	int index = tf_util_insert_index(key, connection->scheduled, connection->scheduled_count, sizeof(tf_ssb_connection_scheduled_t), _tf_ssb_connection_scheduled_compare); | ||||
| 	if (index != connection->scheduled_count && strcmp(key, connection->scheduled[index].key) == 0) | ||||
| 	{ | ||||
| 		tf_ssb_connection_scheduled_t scheduled = connection->scheduled[index]; | ||||
| 		connection->scheduled[index].callback = callback; | ||||
| 		connection->scheduled[index].user_data = user_data; | ||||
|  | ||||
| 		tf_trace_begin(connection->ssb->trace, "scheduled callback (skip)"); | ||||
| 		PRE_CALLBACK(connection->ssb, scheduled.callback); | ||||
| 		scheduled.callback(connection, true, scheduled.user_data); | ||||
| 		POST_CALLBACK(connection->ssb, scheduled.callback); | ||||
| 		tf_trace_end(connection->ssb->trace); | ||||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| 		connection->scheduled = tf_resize_vec(connection->scheduled, sizeof(tf_ssb_connection_scheduled_t) * (connection->scheduled_count + 1)); | ||||
| 	connection->scheduled[connection->scheduled_count++] = (tf_ssb_connection_scheduled_t) { | ||||
| 		memmove(connection->scheduled + index + 1, connection->scheduled + index, sizeof(tf_ssb_connection_scheduled_t) * (connection->scheduled_count - index)); | ||||
| 		connection->scheduled[index] = (tf_ssb_connection_scheduled_t) { | ||||
| 			.callback = callback, | ||||
| 			.user_data = user_data, | ||||
| 		}; | ||||
| 		snprintf(connection->scheduled[index].key, sizeof(connection->scheduled[index].key), "%s", key); | ||||
| 		connection->scheduled_count++; | ||||
| 	} | ||||
|  | ||||
| 	uv_async_send(&connection->scheduled_async); | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -799,17 +799,19 @@ JSValue tf_ssb_connection_requests_to_object(tf_ssb_connection_t* connection); | ||||
| /** | ||||
| ** A function scheduled to be run later. | ||||
| ** @param connection The owning connection. | ||||
| ** @param skip Whether the work ought to be skipped, because it is being replaced. | ||||
| ** @param user_data User data registered with the callback. | ||||
| */ | ||||
| typedef void(tf_ssb_scheduled_callback_t)(tf_ssb_connection_t* connection, void* user_data); | ||||
| typedef void(tf_ssb_scheduled_callback_t)(tf_ssb_connection_t* connection, bool skip, void* user_data); | ||||
|  | ||||
| /** | ||||
| ** Schedule work to be run when the connection is next idle. | ||||
| ** @param connection The owning connection. | ||||
| ** @param key A key identifying the work.  If work by the same key already exists, it will be replaced. | ||||
| ** @param callback The callback to call. | ||||
| ** @param user_data User data to pass to the callback. | ||||
| */ | ||||
| void tf_ssb_connection_schedule_idle(tf_ssb_connection_t* connection, tf_ssb_scheduled_callback_t* callback, void* user_data); | ||||
| void tf_ssb_connection_schedule_idle(tf_ssb_connection_t* connection, const char* key, tf_ssb_scheduled_callback_t* callback, void* user_data); | ||||
|  | ||||
| /** | ||||
| ** Schedule work to run on a worker thread. | ||||
|   | ||||
| @@ -855,6 +855,16 @@ static void _tf_ssb_connection_send_history_stream_work(tf_ssb_connection_t* con | ||||
| 	request->out_finished = request->out_max_sequence_seen != request->sequence + k_max - 1; | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_connection_send_history_stream_destroy(tf_ssb_connection_send_history_stream_t* request) | ||||
| { | ||||
| 	for (int i = 0; i < request->out_messages_count; i++) | ||||
| 	{ | ||||
| 		tf_free(request->out_messages[i]); | ||||
| 	} | ||||
| 	tf_free(request->out_messages); | ||||
| 	tf_free(request); | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_connection_send_history_stream_after_work(tf_ssb_connection_t* connection, int result, void* user_data) | ||||
| { | ||||
| 	tf_ssb_connection_send_history_stream_t* request = user_data; | ||||
| @@ -879,24 +889,19 @@ static void _tf_ssb_connection_send_history_stream_after_work(tf_ssb_connection_ | ||||
| 			tf_ssb_connection_rpc_send(connection, k_ssb_rpc_flag_json, request->request_number, NULL, (const uint8_t*)"false", strlen("false"), NULL, NULL, NULL); | ||||
| 		} | ||||
| 	} | ||||
| 	for (int i = 0; i < request->out_messages_count; i++) | ||||
| 	{ | ||||
| 		tf_free(request->out_messages[i]); | ||||
| 	} | ||||
| 	tf_free(request->out_messages); | ||||
| 	tf_free(request); | ||||
| 	_tf_ssb_connection_send_history_stream_destroy(request); | ||||
| } | ||||
|  | ||||
| static void _tf_ssb_connection_send_history_stream_callback(tf_ssb_connection_t* connection, void* user_data) | ||||
| static void _tf_ssb_connection_send_history_stream_callback(tf_ssb_connection_t* connection, bool skip, void* user_data) | ||||
| { | ||||
| 	tf_ssb_connection_adjust_write_count(connection, 1); | ||||
| 	if (tf_ssb_connection_is_connected(connection) && !tf_ssb_is_shutting_down(tf_ssb_connection_get_ssb(connection))) | ||||
| 	if (!skip && tf_ssb_connection_is_connected(connection) && !tf_ssb_is_shutting_down(tf_ssb_connection_get_ssb(connection))) | ||||
| 	{ | ||||
| 		tf_ssb_connection_run_work(connection, _tf_ssb_connection_send_history_stream_work, _tf_ssb_connection_send_history_stream_after_work, user_data); | ||||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| 		_tf_ssb_connection_send_history_stream_after_work(connection, -1, user_data); | ||||
| 		_tf_ssb_connection_send_history_stream_destroy(user_data); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -914,7 +919,7 @@ static void _tf_ssb_connection_send_history_stream( | ||||
| 			.end_request = end_request, | ||||
| 		}; | ||||
| 		snprintf(async->author, sizeof(async->author), "%s", author); | ||||
| 		tf_ssb_connection_schedule_idle(connection, _tf_ssb_connection_send_history_stream_callback, async); | ||||
| 		tf_ssb_connection_schedule_idle(connection, author, _tf_ssb_connection_send_history_stream_callback, async); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1166,11 +1171,14 @@ typedef struct _resend_clock_t | ||||
| 	int32_t request_number; | ||||
| } resend_clock_t; | ||||
|  | ||||
| static void _tf_ssb_rpc_ebt_replicate_resend_clock(tf_ssb_connection_t* connection, void* user_data) | ||||
| static void _tf_ssb_rpc_ebt_replicate_resend_clock(tf_ssb_connection_t* connection, bool skip, void* user_data) | ||||
| { | ||||
| 	resend_clock_t* resend = user_data; | ||||
| 	if (!skip) | ||||
| 	{ | ||||
| 		_tf_ssb_rpc_ebt_replicate_send_clock(resend->connection, resend->request_number, JS_UNDEFINED); | ||||
| 		tf_ssb_connection_set_sent_clock(resend->connection, true); | ||||
| 	} | ||||
| 	tf_free(user_data); | ||||
| } | ||||
|  | ||||
| @@ -1208,7 +1216,7 @@ static void _tf_ssb_rpc_ebt_replicate(tf_ssb_connection_t* connection, uint8_t f | ||||
| 				.connection = connection, | ||||
| 				.request_number = request_number, | ||||
| 			}; | ||||
| 			tf_ssb_connection_schedule_idle(connection, _tf_ssb_rpc_ebt_replicate_resend_clock, resend); | ||||
| 			tf_ssb_connection_schedule_idle(connection, "ebt.clock", _tf_ssb_rpc_ebt_replicate_resend_clock, resend); | ||||
| 		} | ||||
| 	} | ||||
| 	else | ||||
|   | ||||
		Reference in New Issue
	
	Block a user