forked from cory/tildefriends
		
	Took a whack at cleaning up old blobs.
git-svn-id: https://www.unprompted.com/svn/projects/tildefriends/trunk@4369 ed5197a5-7fde-0310-b194-c3ffbd925b24
This commit is contained in:
		
							
								
								
									
										10
									
								
								core/core.js
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								core/core.js
									
									
									
									
									
								
							@@ -72,6 +72,16 @@ const k_global_settings = {
 | 
			
		||||
		default_value: undefined,
 | 
			
		||||
		description: 'Comma-separated list of host names to which HTTP fetch requests are allowed.  None if empty.',
 | 
			
		||||
	},
 | 
			
		||||
	blob_fetch_age_seconds: {
 | 
			
		||||
		type: 'integer',
 | 
			
		||||
		default_value: undefined,
 | 
			
		||||
		description: 'Only blobs mentioned more recently than this age will be automatically fetched.',
 | 
			
		||||
	},
 | 
			
		||||
	blob_expire_age_seconds: {
 | 
			
		||||
		type: 'integer',
 | 
			
		||||
		default_value: undefined,
 | 
			
		||||
		description: 'Blobs older than this will be automatically deleted.',
 | 
			
		||||
	},
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
let gGlobalSettings = {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,8 +1,8 @@
 | 
			
		||||
<?xml version="1.0" encoding="utf-8"?>
 | 
			
		||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
 | 
			
		||||
	package="com.unprompted.tildefriends"
 | 
			
		||||
	versionCode="9"
 | 
			
		||||
	versionName="0.0.9">
 | 
			
		||||
	versionCode="10"
 | 
			
		||||
	versionName="0.0.10-wip">
 | 
			
		||||
	<uses-sdk android:minSdkVersion="26"/>
 | 
			
		||||
	<uses-permission android:name="android.permission.INTERNET"/>
 | 
			
		||||
	<application android:label="Tilde Friends" android:usesCleartextTraffic="true" android:debuggable="true">
 | 
			
		||||
 
 | 
			
		||||
@@ -184,9 +184,10 @@ void tf_ssb_db_init(tf_ssb_t* ssb)
 | 
			
		||||
	_tf_ssb_db_exec(db, "CREATE INDEX IF NOT EXISTS messages_refs_ref_idx ON messages_refs (ref)");
 | 
			
		||||
	_tf_ssb_db_exec(db, "DROP VIEW IF EXISTS blob_wants_view");
 | 
			
		||||
	_tf_ssb_db_exec(db,
 | 
			
		||||
		"CREATE VIEW IF NOT EXISTS blob_wants_view (id) AS "
 | 
			
		||||
		"  SELECT messages_refs.ref AS id "
 | 
			
		||||
		"CREATE VIEW IF NOT EXISTS blob_wants_view (id, timestamp) AS "
 | 
			
		||||
		"  SELECT messages_refs.ref AS id, messages.timestamp AS timestamp "
 | 
			
		||||
		"  FROM messages_refs "
 | 
			
		||||
		"  JOIN messages ON messages.id = messages_refs.message "
 | 
			
		||||
		"  LEFT OUTER JOIN blobs ON messages_refs.ref = blobs.id "
 | 
			
		||||
		"  WHERE blobs.id IS NULL "
 | 
			
		||||
		"    AND LENGTH(messages_refs.ref) = 52 "
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										236
									
								
								src/ssb.rpc.c
									
									
									
									
									
								
							
							
						
						
									
										236
									
								
								src/ssb.rpc.c
									
									
									
									
									
								
							@@ -10,6 +10,7 @@
 | 
			
		||||
#include "uv.h"
 | 
			
		||||
 | 
			
		||||
#include <inttypes.h>
 | 
			
		||||
#include <stdlib.h>
 | 
			
		||||
#include <string.h>
 | 
			
		||||
#include <time.h>
 | 
			
		||||
 | 
			
		||||
@@ -19,6 +20,80 @@
 | 
			
		||||
 | 
			
		||||
static void _tf_ssb_connection_send_history_stream(tf_ssb_connection_t* connection, int32_t request_number, const char* author, int64_t sequence, bool keys, bool live);
 | 
			
		||||
static void _tf_ssb_connection_send_history_stream_internal(tf_ssb_connection_t* connection, int32_t request_number, const char* author, int64_t sequence, bool keys, bool live);
 | 
			
		||||
static void _tf_ssb_rpc_start_delete_blobs(tf_ssb_t* ssb, int delay_ms);
 | 
			
		||||
 | 
			
		||||
static int64_t _get_global_setting_int64(tf_ssb_t* ssb, const char* name, int64_t default_value)
 | 
			
		||||
{
 | 
			
		||||
	int64_t result = default_value;
 | 
			
		||||
	sqlite3* db = tf_ssb_acquire_db_reader(ssb);
 | 
			
		||||
	sqlite3_stmt* statement;
 | 
			
		||||
	if (sqlite3_prepare(db, "SELECT json_extract(value, '$.' || ?) FROM properties WHERE id = 'core' AND key = 'settings'", -1, &statement, NULL) == SQLITE_OK)
 | 
			
		||||
	{
 | 
			
		||||
		if (sqlite3_bind_text(statement, 1, name, -1, NULL) == SQLITE_OK)
 | 
			
		||||
		{
 | 
			
		||||
			if (sqlite3_step(statement) == SQLITE_ROW)
 | 
			
		||||
			{
 | 
			
		||||
				result = sqlite3_column_int64(statement, 0);
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		sqlite3_finalize(statement);
 | 
			
		||||
	}
 | 
			
		||||
	else
 | 
			
		||||
	{
 | 
			
		||||
		tf_printf("prepare failed: %s\n", sqlite3_errmsg(db));
 | 
			
		||||
	}
 | 
			
		||||
	tf_ssb_release_db_reader(ssb, db);
 | 
			
		||||
	return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static bool _get_global_setting_bool(tf_ssb_t* ssb, const char* name, bool default_value)
 | 
			
		||||
{
 | 
			
		||||
	bool result = default_value;
 | 
			
		||||
	sqlite3* db = tf_ssb_acquire_db_reader(ssb);
 | 
			
		||||
	sqlite3_stmt* statement;
 | 
			
		||||
	if (sqlite3_prepare(db, "SELECT json_extract(value, '$.' || ?) FROM properties WHERE id = 'core' AND key = 'settings'", -1, &statement, NULL) == SQLITE_OK)
 | 
			
		||||
	{
 | 
			
		||||
		if (sqlite3_bind_text(statement, 1, name, -1, NULL) == SQLITE_OK)
 | 
			
		||||
		{
 | 
			
		||||
			if (sqlite3_step(statement) == SQLITE_ROW)
 | 
			
		||||
			{
 | 
			
		||||
				result = sqlite3_column_int(statement, 0) != 0;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		sqlite3_finalize(statement);
 | 
			
		||||
	}
 | 
			
		||||
	else
 | 
			
		||||
	{
 | 
			
		||||
		tf_printf("prepare failed: %s\n", sqlite3_errmsg(db));
 | 
			
		||||
	}
 | 
			
		||||
	tf_ssb_release_db_reader(ssb, db);
 | 
			
		||||
	return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static bool _get_global_setting_string(tf_ssb_t* ssb, const char* name, char* out_value, size_t size)
 | 
			
		||||
{
 | 
			
		||||
	bool result = false;
 | 
			
		||||
	sqlite3* db = tf_ssb_acquire_db_reader(ssb);
 | 
			
		||||
	sqlite3_stmt* statement;
 | 
			
		||||
	if (sqlite3_prepare(db, "SELECT json_extract(value, '$.' || ?) FROM properties WHERE id = 'core' AND key = 'settings'", -1, &statement, NULL) == SQLITE_OK)
 | 
			
		||||
	{
 | 
			
		||||
		if (sqlite3_bind_text(statement, 1, name, -1, NULL) == SQLITE_OK)
 | 
			
		||||
		{
 | 
			
		||||
			if (sqlite3_step(statement) == SQLITE_ROW)
 | 
			
		||||
			{
 | 
			
		||||
				snprintf(out_value, size, "%s", sqlite3_column_text(statement, 0));
 | 
			
		||||
				result = true;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		sqlite3_finalize(statement);
 | 
			
		||||
	}
 | 
			
		||||
	else
 | 
			
		||||
	{
 | 
			
		||||
		tf_printf("prepare failed: %s\n", sqlite3_errmsg(db));
 | 
			
		||||
	}
 | 
			
		||||
	tf_ssb_release_db_reader(ssb, db);
 | 
			
		||||
	return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void _tf_ssb_rpc_gossip_ping_callback(tf_ssb_connection_t* connection, uint8_t flags, int32_t request_number, JSValue args, const uint8_t* message, size_t size, void* user_data)
 | 
			
		||||
{
 | 
			
		||||
@@ -143,11 +218,25 @@ static void _tf_ssb_rpc_request_more_blobs(tf_ssb_connection_t* connection)
 | 
			
		||||
	JSContext* context = tf_ssb_connection_get_context(connection);
 | 
			
		||||
	tf_ssb_blob_wants_t* blob_wants = tf_ssb_connection_get_blob_wants_state(connection);
 | 
			
		||||
	tf_ssb_t* ssb = tf_ssb_connection_get_ssb(connection);
 | 
			
		||||
	int64_t age = _get_global_setting_int64(ssb, "blob_fetch_age_seconds", -1);
 | 
			
		||||
	int64_t timestamp = -1;
 | 
			
		||||
	if (age == 0)
 | 
			
		||||
	{
 | 
			
		||||
		/* Don't fetch any blobs. */
 | 
			
		||||
		return;
 | 
			
		||||
	}
 | 
			
		||||
	else if (age > 0)
 | 
			
		||||
	{
 | 
			
		||||
		int64_t now = (int64_t)time(NULL) * 1000ULL;
 | 
			
		||||
		timestamp = now - age * 1000ULL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	sqlite3* db = tf_ssb_acquire_db_reader(ssb);
 | 
			
		||||
	sqlite3_stmt* statement;
 | 
			
		||||
	if (sqlite3_prepare(db, "SELECT id FROM blob_wants_view WHERE id > ? ORDER BY id LIMIT 32", -1, &statement, NULL) == SQLITE_OK)
 | 
			
		||||
	if (sqlite3_prepare(db, "SELECT id FROM blob_wants_view WHERE id > ? AND timestamp > ? ORDER BY id LIMIT 32", -1, &statement, NULL) == SQLITE_OK)
 | 
			
		||||
	{
 | 
			
		||||
		if (sqlite3_bind_text(statement, 1, blob_wants->last_id, -1, NULL) == SQLITE_OK)
 | 
			
		||||
		if (sqlite3_bind_text(statement, 1, blob_wants->last_id, -1, NULL) == SQLITE_OK &&
 | 
			
		||||
			sqlite3_bind_int64(statement, 2, timestamp) == SQLITE_OK)
 | 
			
		||||
		{
 | 
			
		||||
			while (sqlite3_step(statement) == SQLITE_ROW)
 | 
			
		||||
			{
 | 
			
		||||
@@ -195,67 +284,6 @@ void _tf_ssb_rpc_tunnel_cleanup(tf_ssb_t* ssb, void* user_data)
 | 
			
		||||
	tf_free(user_data);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static bool _get_global_setting_bool(tf_ssb_t* ssb, const char* name, bool default_value)
 | 
			
		||||
{
 | 
			
		||||
	bool result = default_value;
 | 
			
		||||
	JSContext* context = tf_ssb_get_context(ssb);
 | 
			
		||||
	sqlite3* db = tf_ssb_acquire_db_reader(ssb);
 | 
			
		||||
	sqlite3_stmt* statement;
 | 
			
		||||
	if (sqlite3_prepare(db, "SELECT value FROM properties WHERE id = 'core' AND key = 'settings'", -1, &statement, NULL) == SQLITE_OK)
 | 
			
		||||
	{
 | 
			
		||||
		if (sqlite3_step(statement) == SQLITE_ROW)
 | 
			
		||||
		{
 | 
			
		||||
			JSValue value = JS_ParseJSON(context, (const char*)sqlite3_column_text(statement, 0), sqlite3_column_bytes(statement, 0), NULL);
 | 
			
		||||
			JSValue property = JS_GetPropertyStr(context, value,  name);
 | 
			
		||||
			if (JS_IsBool(property))
 | 
			
		||||
			{
 | 
			
		||||
				result = JS_ToBool(context, property);
 | 
			
		||||
			}
 | 
			
		||||
			JS_FreeValue(context, property);
 | 
			
		||||
			JS_FreeValue(context, value);
 | 
			
		||||
		}
 | 
			
		||||
		sqlite3_finalize(statement);
 | 
			
		||||
	}
 | 
			
		||||
	else
 | 
			
		||||
	{
 | 
			
		||||
		tf_printf("prepare failed: %s\n", sqlite3_errmsg(db));
 | 
			
		||||
	}
 | 
			
		||||
	tf_ssb_release_db_reader(ssb, db);
 | 
			
		||||
	return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static bool _get_global_setting_string(tf_ssb_t* ssb, const char* name, char* out_value, size_t size)
 | 
			
		||||
{
 | 
			
		||||
	bool result = false;
 | 
			
		||||
	JSContext* context = tf_ssb_get_context(ssb);
 | 
			
		||||
	sqlite3* db = tf_ssb_acquire_db_reader(ssb);
 | 
			
		||||
	sqlite3_stmt* statement;
 | 
			
		||||
	if (sqlite3_prepare(db, "SELECT value FROM properties WHERE id = 'core' AND key = 'settings'", -1, &statement, NULL) == SQLITE_OK)
 | 
			
		||||
	{
 | 
			
		||||
		if (sqlite3_step(statement) == SQLITE_ROW)
 | 
			
		||||
		{
 | 
			
		||||
			JSValue value = JS_ParseJSON(context, (const char*)sqlite3_column_text(statement, 0), sqlite3_column_bytes(statement, 0), NULL);
 | 
			
		||||
			JSValue property = JS_GetPropertyStr(context, value, name);
 | 
			
		||||
			const char* value_string = JS_ToCString(context, property);
 | 
			
		||||
			if (value_string)
 | 
			
		||||
			{
 | 
			
		||||
				snprintf(out_value, size, "%s", value_string);
 | 
			
		||||
				result = true;
 | 
			
		||||
				JS_FreeCString(context, value_string);
 | 
			
		||||
			}
 | 
			
		||||
			JS_FreeValue(context, property);
 | 
			
		||||
			JS_FreeValue(context, value);
 | 
			
		||||
		}
 | 
			
		||||
		sqlite3_finalize(statement);
 | 
			
		||||
	}
 | 
			
		||||
	else
 | 
			
		||||
	{
 | 
			
		||||
		tf_printf("prepare failed: %s\n", sqlite3_errmsg(db));
 | 
			
		||||
	}
 | 
			
		||||
	tf_ssb_release_db_reader(ssb, db);
 | 
			
		||||
	return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void _tf_ssb_rpc_tunnel_connect(tf_ssb_connection_t* connection, uint8_t flags, int32_t request_number, JSValue args, const uint8_t* message, size_t size, void* user_data)
 | 
			
		||||
{
 | 
			
		||||
	tf_ssb_t* ssb = tf_ssb_connection_get_ssb(connection);
 | 
			
		||||
@@ -1116,6 +1144,87 @@ static void _tf_ssb_rpc_connections_changed_callback(tf_ssb_t* ssb, tf_ssb_chang
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void _tf_ssb_rpc_delete_blobs_work(uv_work_t* work)
 | 
			
		||||
{
 | 
			
		||||
	tf_ssb_t* ssb = work->data;
 | 
			
		||||
	int64_t age = _get_global_setting_int64(ssb, "blob_expire_age_seconds", -1);
 | 
			
		||||
	if (age <= 0)
 | 
			
		||||
	{
 | 
			
		||||
		return;
 | 
			
		||||
	}
 | 
			
		||||
	sqlite3* db = tf_ssb_acquire_db_writer(ssb);
 | 
			
		||||
	sqlite3_stmt* statement;
 | 
			
		||||
	int64_t now = (int64_t)time(NULL) * 1000ULL;
 | 
			
		||||
	int64_t timestamp = now - age * 1000ULL;
 | 
			
		||||
	const int k_limit = 256;
 | 
			
		||||
	bool deleted = false;
 | 
			
		||||
	if (sqlite3_prepare(db,
 | 
			
		||||
		"DELETE FROM blobs WHERE blobs.id IN ("
 | 
			
		||||
		"  SELECT blobs.id FROM blobs "
 | 
			
		||||
		"  JOIN messages_refs ON blobs.id = messages_refs.ref "
 | 
			
		||||
		"  JOIN messages ON messages.id = messages_refs.message "
 | 
			
		||||
		"  GROUP BY blobs.id HAVING MAX(messages.timestamp) < ? LIMIT ?)", -1, &statement, NULL) == SQLITE_OK)
 | 
			
		||||
	{
 | 
			
		||||
		if (sqlite3_bind_int64(statement, 1, timestamp) == SQLITE_OK &&
 | 
			
		||||
			sqlite3_bind_int(statement, 2, k_limit) == SQLITE_OK)
 | 
			
		||||
		{
 | 
			
		||||
			int r = sqlite3_step(statement);
 | 
			
		||||
			if (r != SQLITE_DONE)
 | 
			
		||||
			{
 | 
			
		||||
				tf_printf("_tf_ssb_rpc_delete_blobs_work: %s\n", sqlite3_errmsg(db));
 | 
			
		||||
			}
 | 
			
		||||
			else
 | 
			
		||||
			{
 | 
			
		||||
				tf_printf("_tf_ssb_rpc_delete_blobs_work: %d rows\n", sqlite3_changes(db));
 | 
			
		||||
			}
 | 
			
		||||
			deleted = sqlite3_changes(db) != 0;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	else
 | 
			
		||||
	{
 | 
			
		||||
		tf_printf("prepare failed: %s\n", sqlite3_errmsg(db));
 | 
			
		||||
	}
 | 
			
		||||
	tf_ssb_release_db_writer(ssb, db);
 | 
			
		||||
 | 
			
		||||
	if (deleted)
 | 
			
		||||
	{
 | 
			
		||||
		_tf_ssb_rpc_start_delete_blobs(ssb, 60 * 1000);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void _tf_ssb_rpc_delete_blobs_after_work(uv_work_t* work, int status)
 | 
			
		||||
{
 | 
			
		||||
	tf_free(work);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void _tf_ssb_rpc_timer_on_close(uv_handle_t* handle)
 | 
			
		||||
{
 | 
			
		||||
	tf_free(handle);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void _tf_ssb_rpc_start_delete_timer(uv_timer_t* timer)
 | 
			
		||||
{
 | 
			
		||||
	tf_ssb_t* ssb = timer->data;
 | 
			
		||||
	uv_work_t* work = tf_malloc(sizeof(uv_work_t));
 | 
			
		||||
	*work = (uv_work_t) { .data = ssb };
 | 
			
		||||
	int r = uv_queue_work(tf_ssb_get_loop(ssb), work, _tf_ssb_rpc_delete_blobs_work, _tf_ssb_rpc_delete_blobs_after_work);
 | 
			
		||||
	if (r)
 | 
			
		||||
	{
 | 
			
		||||
		tf_printf("uv_queue_work: %s\n", uv_strerror(r));
 | 
			
		||||
		tf_free(work);
 | 
			
		||||
	}
 | 
			
		||||
	uv_close((uv_handle_t*)timer, _tf_ssb_rpc_timer_on_close);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void _tf_ssb_rpc_start_delete_blobs(tf_ssb_t* ssb, int delay_ms)
 | 
			
		||||
{
 | 
			
		||||
	tf_printf("will delete more blobs in %d ms\n", delay_ms);
 | 
			
		||||
	uv_timer_t* timer = tf_malloc(sizeof(uv_timer_t));
 | 
			
		||||
	*timer = (uv_timer_t) { .data = ssb };
 | 
			
		||||
	uv_timer_init(tf_ssb_get_loop(ssb), timer);
 | 
			
		||||
	uv_timer_start(timer, _tf_ssb_rpc_start_delete_timer, delay_ms, 0);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void tf_ssb_rpc_register(tf_ssb_t* ssb)
 | 
			
		||||
{
 | 
			
		||||
	tf_ssb_add_connections_changed_callback(ssb, _tf_ssb_rpc_connections_changed_callback, NULL, NULL);
 | 
			
		||||
@@ -1129,4 +1238,5 @@ void tf_ssb_rpc_register(tf_ssb_t* ssb)
 | 
			
		||||
	tf_ssb_add_rpc_callback(ssb, (const char*[]) { "room", "attendants", NULL }, _tf_ssb_rpc_room_attendants, NULL, NULL); /* SOURCE */
 | 
			
		||||
	tf_ssb_add_rpc_callback(ssb, (const char*[]) { "createHistoryStream", NULL }, _tf_ssb_rpc_createHistoryStream, NULL, NULL); /* SOURCE */
 | 
			
		||||
	tf_ssb_add_rpc_callback(ssb, (const char*[]) { "ebt", "replicate", NULL }, _tf_ssb_rpc_ebt_replicate_server, NULL, NULL); /* DUPLEX */
 | 
			
		||||
	_tf_ssb_rpc_start_delete_blobs(ssb, 0);
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user