Thu, 24 Jul 2025 23:35:13 -0500
Create Purple.Scheduler
This manages Purple.ScheduledTask instances and propagates their execute
signals.
This can be used for stuff like reconnecting accounts, scheduling messages
regardless of protocol support, changing avatars at a set time, really
whatever you can think of and write a plugin to implement.
Testing Done:
Ran the unit tests under valgrind and called in the turtles.
Also run in a dev environment and verified that there were no weird error messages.
Bugs closed: PIDGIN-18105
Reviewed at https://reviews.imfreedom.org/r/4073/
| 43293 | 1 | /* |
| 2 | * Purple - Internet Messaging Library | |
| 3 | * Copyright (C) Pidgin Developers <devel@pidgin.im> | |
| 4 | * | |
| 5 | * Purple is the legal property of its developers, whose names are too numerous | |
| 6 | * to list here. Please refer to the COPYRIGHT file distributed with this | |
| 7 | * source distribution. | |
| 8 | * | |
| 9 | * This library is free software; you can redistribute it and/or modify it | |
| 10 | * under the terms of the GNU General Public License as published by the Free | |
| 11 | * Software Foundation; either version 2 of the License, or (at your option) | |
| 12 | * any later version. | |
| 13 | * | |
| 14 | * This library is distributed in the hope that it will be useful, but WITHOUT | |
| 15 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | |
| 16 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | |
| 17 | * more details. | |
| 18 | * | |
| 19 | * You should have received a copy of the GNU General Public License along with | |
| 20 | * this library; if not, see <https://www.gnu.org/licenses/>. | |
| 21 | */ | |
| 22 | ||
| 23 | #ifdef G_LOG_DOMAIN | |
| 24 | # undef G_LOG_DOMAIN | |
| 25 | #endif | |
| 26 | #define G_LOG_DOMAIN "PurpleScheduler" | |
| 27 | ||
| 28 | #include <birb.h> | |
| 29 | ||
| 30 | #include "purplescheduler.h" | |
| 31 | #include "purpleschedulerprivate.h" | |
| 32 | ||
| 33 | struct _PurpleScheduler { | |
| 34 | GObject parent; | |
| 35 | ||
| 36 | GPtrArray *tasks; | |
| 37 | }; | |
| 38 | ||
| 39 | enum { | |
| 40 | PROP_0, | |
| 41 | PROP_ITEM_TYPE, | |
| 42 | PROP_N_ITEMS, | |
| 43 | N_PROPERTIES, | |
| 44 | }; | |
| 45 | static GParamSpec *properties[N_PROPERTIES] = {NULL, }; | |
| 46 | ||
| 47 | enum { | |
| 48 | SIG_EXECUTE_TASK, | |
| 49 | N_SIGNALS, | |
| 50 | }; | |
| 51 | static guint signals[N_SIGNALS] = {0, }; | |
| 52 | ||
| 53 | G_DEFINE_QUARK(purple-scheduler-error, purple_scheduler_error) | |
| 54 | ||
| 55 | static PurpleScheduler *default_scheduler = NULL; | |
| 56 | ||
| 57 | /****************************************************************************** | |
| 58 | * Helpers | |
| 59 | *****************************************************************************/ | |
| 60 | /** | |
| 61 | * purple_scheduler_find_task_with_id: (skip) | |
| 62 | * @id: the id of the task to search for | |
| 63 | * @position: (out) (nullable): a return address for the position of the item | |
| 64 | * | |
| 65 | * Looks for a task with the given id. | |
| 66 | * | |
| 67 | * If the task is found it will be returned as well as it's position. | |
| 68 | * | |
| 69 | * Returns: (transfer none) (nullable): The task if found. | |
| 70 | * | |
| 71 | * Since: 3.0 | |
| 72 | */ | |
| 73 | static PurpleScheduledTask * | |
| 74 | purple_scheduler_find_task_with_id(PurpleScheduler *scheduler, | |
| 75 | const char *id, | |
| 76 | guint *position) | |
| 77 | { | |
| 78 | g_return_val_if_fail(PURPLE_IS_SCHEDULER(scheduler), NULL); | |
| 79 | ||
| 80 | for(guint i = 0; i < scheduler->tasks->len; i++) { | |
| 81 | PurpleScheduledTask *task = NULL; | |
| 82 | const char *task_id = NULL; | |
| 83 | ||
| 84 | task = g_ptr_array_index(scheduler->tasks, i); | |
| 85 | task_id = purple_scheduled_task_get_id(task); | |
| 86 | ||
| 87 | if(birb_str_equal(id, task_id)) { | |
| 88 | if(position != NULL) { | |
| 89 | *position = i; | |
| 90 | } | |
| 91 | ||
| 92 | return task; | |
| 93 | } | |
| 94 | } | |
| 95 | ||
| 96 | return NULL; | |
| 97 | } | |
| 98 | ||
| 99 | /** | |
| 100 | * purple_scheduler_unref_task: (skip) | |
| 101 | * @task: the task to cancel and unref | |
| 102 | * | |
| 103 | * Cancels a task if necessary before unreferencing it. | |
| 104 | * | |
| 105 | * Since: 3.0 | |
| 106 | */ | |
| 107 | static void | |
| 108 | purple_scheduler_unref_task(PurpleScheduledTask *task) { | |
| 109 | PurpleScheduledTaskState state = PURPLE_SCHEDULED_TASK_STATE_UNSCHEDULED; | |
| 110 | ||
| 111 | g_return_if_fail(PURPLE_IS_SCHEDULED_TASK(task)); | |
| 112 | ||
| 113 | state = purple_scheduled_task_get_state(task); | |
| 114 | if(state == PURPLE_SCHEDULED_TASK_STATE_SCHEDULED) { | |
| 115 | purple_scheduled_task_cancel(task); | |
| 116 | } | |
| 117 | ||
| 118 | g_object_unref(task); | |
| 119 | } | |
| 120 | ||
| 121 | /****************************************************************************** | |
| 122 | * Callbacks | |
| 123 | *****************************************************************************/ | |
| 124 | static void | |
| 125 | purple_scheduler_task_execute_cb(PurpleScheduledTask *task, gpointer data) { | |
| 126 | PurpleScheduler *scheduler = data; | |
| 127 | const char *task_type = NULL; | |
| 128 | ||
| 129 | task_type = purple_scheduled_task_get_task_type(task); | |
| 130 | g_signal_emit(scheduler, | |
| 131 | signals[SIG_EXECUTE_TASK], | |
| 132 | g_quark_from_string(task_type), | |
| 133 | task, | |
| 134 | task_type); | |
| 135 | } | |
| 136 | ||
| 137 | static void | |
| 138 | purple_scheduler_task_notify_state_cb(GObject *obj, | |
| 139 | G_GNUC_UNUSED GParamSpec *pspec, | |
| 140 | gpointer data) | |
| 141 | { | |
| 142 | PurpleScheduledTask *task = PURPLE_SCHEDULED_TASK(obj); | |
| 143 | PurpleScheduledTaskState state = PURPLE_SCHEDULED_TASK_STATE_UNSCHEDULED; | |
| 144 | PurpleScheduler *scheduler = data; | |
| 145 | ||
| 146 | state = purple_scheduled_task_get_state(task); | |
| 147 | if(state == PURPLE_SCHEDULED_TASK_STATE_EXECUTED) { | |
| 148 | purple_scheduler_remove_task(scheduler, | |
| 149 | purple_scheduled_task_get_id(task)); | |
| 150 | } | |
| 151 | } | |
| 152 | ||
| 153 | /****************************************************************************** | |
| 154 | * GListModel Implementation | |
| 155 | *****************************************************************************/ | |
| 156 | static GType | |
| 157 | purple_scheduler_get_item_type(G_GNUC_UNUSED GListModel *model) { | |
| 158 | return PURPLE_TYPE_SCHEDULED_TASK; | |
| 159 | } | |
| 160 | ||
| 161 | static guint | |
| 162 | purple_scheduler_get_n_items(GListModel *list) { | |
| 163 | PurpleScheduler *scheduler = PURPLE_SCHEDULER(list); | |
| 164 | ||
| 165 | return scheduler->tasks->len; | |
| 166 | } | |
| 167 | ||
| 168 | static gpointer | |
| 169 | purple_scheduler_get_item(GListModel *list, guint position) { | |
| 170 | PurpleScheduler *scheduler = PURPLE_SCHEDULER(list); | |
| 171 | PurpleScheduledTask *task = NULL; | |
| 172 | ||
| 173 | if(position < scheduler->tasks->len) { | |
| 174 | task = g_ptr_array_index(scheduler->tasks, position); | |
| 175 | g_object_ref(task); | |
| 176 | } | |
| 177 | ||
| 178 | return task; | |
| 179 | } | |
| 180 | ||
| 181 | static void | |
| 182 | purple_scheduler_list_model_init(GListModelInterface *iface) { | |
| 183 | iface->get_item_type = purple_scheduler_get_item_type; | |
| 184 | iface->get_n_items = purple_scheduler_get_n_items; | |
| 185 | iface->get_item = purple_scheduler_get_item; | |
| 186 | } | |
| 187 | ||
| 188 | /****************************************************************************** | |
| 189 | * GObject Implementation | |
| 190 | *****************************************************************************/ | |
| 191 | G_DEFINE_FINAL_TYPE_WITH_CODE(PurpleScheduler, purple_scheduler, G_TYPE_OBJECT, | |
| 192 | G_IMPLEMENT_INTERFACE(G_TYPE_LIST_MODEL, | |
| 193 | purple_scheduler_list_model_init)); | |
| 194 | ||
| 195 | static void | |
| 196 | purple_scheduler_finalize(GObject *obj) { | |
| 197 | PurpleScheduler *scheduler = PURPLE_SCHEDULER(obj); | |
| 198 | ||
| 199 | g_clear_pointer(&scheduler->tasks, g_ptr_array_unref); | |
| 200 | ||
| 201 | G_OBJECT_CLASS(purple_scheduler_parent_class)->finalize(obj); | |
| 202 | } | |
| 203 | ||
| 204 | static void | |
| 205 | purple_scheduler_get_property(GObject *obj, guint param_id, GValue *value, | |
| 206 | GParamSpec *pspec) | |
| 207 | { | |
| 208 | GListModel *model = G_LIST_MODEL(obj); | |
| 209 | ||
| 210 | switch(param_id) { | |
| 211 | case PROP_ITEM_TYPE: | |
| 212 | g_value_set_gtype(value, g_list_model_get_item_type(model)); | |
| 213 | break; | |
| 214 | case PROP_N_ITEMS: | |
| 215 | g_value_set_uint(value, g_list_model_get_n_items(model)); | |
| 216 | break; | |
| 217 | default: | |
| 218 | G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec); | |
| 219 | break; | |
| 220 | } | |
| 221 | } | |
| 222 | ||
| 223 | static void | |
| 224 | purple_scheduler_init(PurpleScheduler *scheduler) { | |
| 225 | scheduler->tasks = g_ptr_array_new_full(10, | |
| 226 | (GDestroyNotify)purple_scheduler_unref_task); | |
| 227 | } | |
| 228 | ||
| 229 | static void | |
| 230 | purple_scheduler_class_init(PurpleSchedulerClass *klass) { | |
| 231 | GObjectClass *obj_class = G_OBJECT_CLASS(klass); | |
| 232 | ||
| 233 | obj_class->finalize = purple_scheduler_finalize; | |
| 234 | obj_class->get_property = purple_scheduler_get_property; | |
| 235 | ||
| 236 | /** | |
| 237 | * PurpleScheduler:item-type: | |
| 238 | * | |
| 239 | * The type of items. See [vfunc@Gio.ListModel.get_item_type]. | |
| 240 | * | |
| 241 | * Since: 3.0 | |
| 242 | */ | |
| 243 | properties[PROP_ITEM_TYPE] = g_param_spec_gtype( | |
| 244 | "item-type", NULL, NULL, | |
| 245 | G_TYPE_OBJECT, | |
| 246 | G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); | |
| 247 | ||
| 248 | /** | |
| 249 | * PurpleScheduler:n-items: | |
| 250 | * | |
| 251 | * The number of items. See [vfunc@Gio.ListModel.get_n_items]. | |
| 252 | * | |
| 253 | * Since: 3.0 | |
| 254 | */ | |
| 255 | properties[PROP_N_ITEMS] = g_param_spec_uint( | |
| 256 | "n-items", NULL, NULL, | |
| 257 | 0, G_MAXUINT, 0, | |
| 258 | G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); | |
| 259 | ||
| 260 | g_object_class_install_properties(obj_class, N_PROPERTIES, properties); | |
| 261 | ||
| 262 | /** | |
| 263 | * PurpleScheduler::execute-task: | |
| 264 | * @scheduler: the instance | |
| 265 | * @task: the task being executed | |
| 266 | * @task_type: the task type | |
| 267 | * | |
| 268 | * Emitted when a task is being executed. | |
| 269 | * | |
| 270 | * This signal supports details on [property@ScheduledTask:task-type] to | |
| 271 | * make it easier to listen for specific task types being executed. | |
| 272 | * | |
| 273 | * Since: 3.0 | |
| 274 | */ | |
| 275 | signals[SIG_EXECUTE_TASK] = g_signal_new_class_handler( | |
| 276 | "execute-task", | |
| 277 | G_OBJECT_CLASS_TYPE(klass), | |
| 278 | G_SIGNAL_RUN_LAST | G_SIGNAL_DETAILED, | |
| 279 | NULL, | |
| 280 | NULL, | |
| 281 | NULL, | |
| 282 | NULL, | |
| 283 | G_TYPE_NONE, | |
| 284 | 2, | |
| 285 | PURPLE_TYPE_SCHEDULED_TASK, | |
| 286 | G_TYPE_STRING); | |
| 287 | } | |
| 288 | ||
| 289 | /****************************************************************************** | |
| 290 | * Private API | |
| 291 | *****************************************************************************/ | |
| 292 | void | |
| 293 | purple_scheduler_startup(void) { | |
| 294 | if(!PURPLE_IS_SCHEDULER(default_scheduler)) { | |
| 295 | default_scheduler = purple_scheduler_new(); | |
| 296 | g_object_add_weak_pointer(G_OBJECT(default_scheduler), | |
| 297 | (gpointer *)&default_scheduler); | |
| 298 | } | |
| 299 | } | |
| 300 | ||
| 301 | void | |
| 302 | purple_scheduler_shutdown(void) { | |
| 303 | g_clear_object(&default_scheduler); | |
| 304 | } | |
| 305 | ||
| 306 | /****************************************************************************** | |
| 307 | * Public API | |
| 308 | *****************************************************************************/ | |
| 309 | gboolean | |
| 310 | purple_scheduler_add_task(PurpleScheduler *scheduler, | |
| 311 | PurpleScheduledTask *task, | |
| 312 | GDateTime *execute_at, | |
| 313 | GError **error) | |
| 314 | { | |
| 315 | PurpleScheduledTask *existing = NULL; | |
| 316 | GError *local_error = NULL; | |
| 317 | const char *id = NULL; | |
| 318 | ||
| 319 | g_return_val_if_fail(PURPLE_IS_SCHEDULER(scheduler), FALSE); | |
| 320 | g_return_val_if_fail(PURPLE_IS_SCHEDULED_TASK(task), FALSE); | |
| 321 | g_return_val_if_fail(execute_at != NULL, FALSE); | |
| 322 | ||
| 323 | id = purple_scheduled_task_get_id(task); | |
| 324 | existing = purple_scheduler_find_task_with_id(scheduler, id, NULL); | |
| 325 | if(PURPLE_IS_SCHEDULED_TASK(existing)) { | |
| 326 | g_set_error(error, | |
| 327 | PURPLE_SCHEDULER_ERROR, | |
| 328 | PURPLE_SCHEDULER_ERROR_TASK_EXISTS, | |
| 329 | "a task with id %s already exists", | |
| 330 | id); | |
| 331 | return FALSE; | |
| 332 | } | |
| 333 | ||
| 334 | if(!purple_scheduled_task_schedule(task, execute_at, &local_error)) { | |
| 335 | if(local_error != NULL) { | |
| 336 | g_propagate_error(error, local_error); | |
| 337 | } else { | |
| 338 | g_set_error_literal(error, | |
| 339 | PURPLE_SCHEDULER_ERROR, | |
| 340 | PURPLE_SCHEDULER_ERROR_FAILED_TO_SCHEDULE, | |
| 341 | "the task failed to schedule for an unknown " | |
| 342 | "reason"); | |
| 343 | } | |
| 344 | ||
| 345 | return FALSE; | |
| 346 | } | |
| 347 | ||
| 348 | /* Connect to the execute signal so we can propagate the signal. */ | |
| 349 | g_signal_connect_object(task, | |
| 350 | "execute", | |
| 351 | G_CALLBACK(purple_scheduler_task_execute_cb), | |
| 352 | scheduler, | |
| 353 | G_CONNECT_DEFAULT); | |
| 354 | ||
| 355 | /* Add a handler to remove the task after it's been executed. */ | |
| 356 | g_signal_connect_object(task, | |
| 357 | "notify::state", | |
| 358 | G_CALLBACK(purple_scheduler_task_notify_state_cb), | |
| 359 | scheduler, | |
| 360 | G_CONNECT_DEFAULT); | |
| 361 | ||
| 362 | /* Finally add the item and emit the items-changed signal. */ | |
| 363 | g_ptr_array_add(scheduler->tasks, g_object_ref(task)); | |
| 364 | g_list_model_items_changed(G_LIST_MODEL(scheduler), | |
| 365 | scheduler->tasks->len - 1, 0, 1); | |
| 366 | ||
| 367 | return TRUE; | |
| 368 | } | |
| 369 | ||
| 370 | gboolean | |
| 371 | purple_scheduler_add_task_relative(PurpleScheduler *scheduler, | |
| 372 | PurpleScheduledTask *task, | |
| 373 | GTimeSpan when, | |
| 374 | GError **error) | |
| 375 | { | |
| 376 | GDateTime *now = NULL; | |
| 377 | GDateTime *execute_at = NULL; | |
| 378 | gboolean ret = FALSE; | |
| 379 | ||
| 380 | g_return_val_if_fail(PURPLE_IS_SCHEDULER(scheduler), FALSE); | |
| 381 | g_return_val_if_fail(PURPLE_IS_SCHEDULED_TASK(task), FALSE); | |
| 382 | ||
| 383 | now = g_date_time_new_now_local(); | |
| 384 | execute_at = g_date_time_add(now, when); | |
| 385 | g_date_time_unref(now); | |
| 386 | ||
| 387 | ret = purple_scheduler_add_task(scheduler, task, execute_at, error); | |
| 388 | ||
| 389 | g_date_time_unref(execute_at); | |
| 390 | ||
| 391 | return ret; | |
| 392 | } | |
| 393 | ||
| 394 | PurpleScheduler * | |
| 395 | purple_scheduler_get_default(void) { | |
| 396 | return default_scheduler; | |
| 397 | } | |
| 398 | ||
| 399 | GListModel * | |
| 400 | purple_scheduler_get_default_as_model(void) { | |
| 401 | if(G_IS_LIST_MODEL(default_scheduler)) { | |
| 402 | return G_LIST_MODEL(default_scheduler); | |
| 403 | } | |
| 404 | ||
| 405 | return NULL; | |
| 406 | } | |
| 407 | ||
| 408 | PurpleScheduler * | |
| 409 | purple_scheduler_new(void) { | |
| 410 | return g_object_new(PURPLE_TYPE_SCHEDULER, NULL); | |
| 411 | } | |
| 412 | ||
| 413 | gboolean | |
| 414 | purple_scheduler_remove_task(PurpleScheduler *scheduler, const char *id) { | |
| 415 | PurpleScheduledTask *task = NULL; | |
| 416 | PurpleScheduledTaskState state = PURPLE_SCHEDULED_TASK_STATE_UNSCHEDULED; | |
| 417 | guint position = 0; | |
| 418 | ||
| 419 | g_return_val_if_fail(PURPLE_IS_SCHEDULER(scheduler), FALSE); | |
| 420 | g_return_val_if_fail(!birb_str_is_empty(id), FALSE); | |
| 421 | ||
| 422 | task = purple_scheduler_find_task_with_id(scheduler, id, &position); | |
| 423 | if(!PURPLE_IS_SCHEDULED_TASK(task)) { | |
| 424 | return FALSE; | |
| 425 | } | |
| 426 | ||
| 427 | state = purple_scheduled_task_get_state(task); | |
| 428 | if(state == PURPLE_SCHEDULED_TASK_STATE_SCHEDULED) { | |
| 429 | purple_scheduled_task_cancel(task); | |
| 430 | } | |
| 431 | ||
| 432 | g_ptr_array_remove_index(scheduler->tasks, position); | |
| 433 | ||
| 434 | g_list_model_items_changed(G_LIST_MODEL(scheduler), position, 1, 0); | |
| 435 | ||
| 436 | return TRUE; | |
| 437 | } |