socketservice: add an "active" property

We already have start, stop and is_active methods, but turning it
into a real property is useful for a few reasons:
 - it allows us to bind the property to an UI or a setting
 - it allows us to get notified when the state changes
 - it allows us to instantiate objects directly in the stopped state

https://bugzilla.gnome.org/show_bug.cgi?id=752089
This commit is contained in:
Paolo Borelli 2015-07-10 21:30:28 +02:00
parent a223796d0b
commit e1d44799c0
4 changed files with 226 additions and 29 deletions

View File

@ -63,6 +63,7 @@
#include <gio/gio.h> #include <gio/gio.h>
#include "gsocketlistener.h" #include "gsocketlistener.h"
#include "gsocketconnection.h" #include "gsocketconnection.h"
#include "glibintl.h"
struct _GSocketServicePrivate struct _GSocketServicePrivate
{ {
@ -77,6 +78,12 @@ G_LOCK_DEFINE_STATIC(active);
G_DEFINE_TYPE_WITH_PRIVATE (GSocketService, g_socket_service, G_TYPE_SOCKET_LISTENER) G_DEFINE_TYPE_WITH_PRIVATE (GSocketService, g_socket_service, G_TYPE_SOCKET_LISTENER)
enum
{
PROP_0,
PROP_ACTIVE
};
static void g_socket_service_ready (GObject *object, static void g_socket_service_ready (GObject *object,
GAsyncResult *result, GAsyncResult *result,
gpointer user_data); gpointer user_data);
@ -117,6 +124,90 @@ do_accept (GSocketService *service)
service->priv->outstanding_accept = TRUE; service->priv->outstanding_accept = TRUE;
} }
static gboolean
get_active (GSocketService *service)
{
gboolean active;
G_LOCK (active);
active = service->priv->active;
G_UNLOCK (active);
return active;
}
static void
set_active (GSocketService *service, gboolean active)
{
gboolean notify = FALSE;
active = !!active;
G_LOCK (active);
if (active != service->priv->active)
{
service->priv->active = active;
notify = TRUE;
if (active)
{
if (service->priv->outstanding_accept)
g_cancellable_cancel (service->priv->cancellable);
else
do_accept (service);
}
else
{
if (service->priv->outstanding_accept)
g_cancellable_cancel (service->priv->cancellable);
}
}
G_UNLOCK (active);
if (notify)
g_object_notify (G_OBJECT (service), "active");
}
static void
g_socket_service_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
GSocketService *service = G_SOCKET_SERVICE (object);
switch (prop_id)
{
case PROP_ACTIVE:
g_value_set_boolean (value, get_active (service));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void
g_socket_service_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
GSocketService *service = G_SOCKET_SERVICE (object);
switch (prop_id)
{
case PROP_ACTIVE:
set_active (service, g_value_get_boolean (value));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void static void
g_socket_service_changed (GSocketListener *listener) g_socket_service_changed (GSocketListener *listener)
{ {
@ -151,12 +242,9 @@ g_socket_service_changed (GSocketListener *listener)
gboolean gboolean
g_socket_service_is_active (GSocketService *service) g_socket_service_is_active (GSocketService *service)
{ {
gboolean active; g_return_val_if_fail (G_IS_SOCKET_SERVICE (service), FALSE);
G_LOCK (active); return get_active (service);
active = service->priv->active;
G_UNLOCK (active);
return active;
} }
/** /**
@ -174,19 +262,9 @@ g_socket_service_is_active (GSocketService *service)
void void
g_socket_service_start (GSocketService *service) g_socket_service_start (GSocketService *service)
{ {
G_LOCK (active); g_return_if_fail (G_IS_SOCKET_SERVICE (service));
if (!service->priv->active) set_active (service, TRUE);
{
service->priv->active = TRUE;
if (service->priv->outstanding_accept)
g_cancellable_cancel (service->priv->cancellable);
else
do_accept (service);
}
G_UNLOCK (active);
} }
/** /**
@ -210,20 +288,11 @@ g_socket_service_start (GSocketService *service)
void void
g_socket_service_stop (GSocketService *service) g_socket_service_stop (GSocketService *service)
{ {
G_LOCK (active); g_return_if_fail (G_IS_SOCKET_SERVICE (service));
if (service->priv->active) set_active (service, FALSE);
{
service->priv->active = FALSE;
if (service->priv->outstanding_accept)
g_cancellable_cancel (service->priv->cancellable);
}
G_UNLOCK (active);
} }
static gboolean static gboolean
g_socket_service_incoming (GSocketService *service, g_socket_service_incoming (GSocketService *service,
GSocketConnection *connection, GSocketConnection *connection,
@ -243,6 +312,8 @@ g_socket_service_class_init (GSocketServiceClass *class)
GSocketListenerClass *listener_class = G_SOCKET_LISTENER_CLASS (class); GSocketListenerClass *listener_class = G_SOCKET_LISTENER_CLASS (class);
gobject_class->finalize = g_socket_service_finalize; gobject_class->finalize = g_socket_service_finalize;
gobject_class->set_property = g_socket_service_set_property;
gobject_class->get_property = g_socket_service_get_property;
listener_class->changed = g_socket_service_changed; listener_class->changed = g_socket_service_changed;
class->incoming = g_socket_service_real_incoming; class->incoming = g_socket_service_real_incoming;
@ -271,6 +342,20 @@ g_socket_service_class_init (GSocketServiceClass *class)
g_signal_accumulator_true_handled, NULL, g_signal_accumulator_true_handled, NULL,
NULL, G_TYPE_BOOLEAN, NULL, G_TYPE_BOOLEAN,
2, G_TYPE_SOCKET_CONNECTION, G_TYPE_OBJECT); 2, G_TYPE_SOCKET_CONNECTION, G_TYPE_OBJECT);
/**
* GSocketService:active:
*
* Whether the service is currently accepting connections.
*
* Since: 2.46
*/
g_object_class_install_property (gobject_class, PROP_ACTIVE,
g_param_spec_boolean ("active",
P_("Active"),
P_("Whether the service is currently accepting connections"),
TRUE,
G_PARAM_CONSTRUCT | G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
} }
static void static void
@ -309,7 +394,6 @@ g_socket_service_ready (GObject *object,
G_UNLOCK (active); G_UNLOCK (active);
} }
/** /**
* g_socket_service_new: * g_socket_service_new:
* *

View File

@ -120,6 +120,7 @@ socket
socket-address socket-address
socket-client socket-client
socket-listener socket-listener
socket-service
socket-server socket-server
srvtarget srvtarget
task task

View File

@ -57,6 +57,7 @@ test_programs = \
sleepy-stream \ sleepy-stream \
socket \ socket \
socket-listener \ socket-listener \
socket-service \
srvtarget \ srvtarget \
task \ task \
tls-interaction \ tls-interaction \

111
gio/tests/socket-service.c Normal file
View File

@ -0,0 +1,111 @@
/* GLib testing framework examples and tests
*
* Copyright 2014 Red Hat, Inc.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, see
* <http://www.gnu.org/licenses/>.
*/
#include <gio/gio.h>
static void
active_notify_cb (GSocketService *service,
GParamSpec *pspec,
gpointer data)
{
gboolean *success = (gboolean *)data;
if (g_socket_service_is_active (service))
*success = TRUE;
}
static void
connected_cb (GObject *client,
GAsyncResult *result,
gpointer user_data)
{
GSocketService *service = G_SOCKET_SERVICE (user_data);
GSocketConnection *conn;
GError *error = NULL;
g_assert_true (g_socket_service_is_active (service));
conn = g_socket_client_connect_finish (G_SOCKET_CLIENT (client), result, &error);
g_assert_no_error (error);
g_object_unref (conn);
g_socket_service_stop (service);
g_assert_false (g_socket_service_is_active (service));
}
static void
test_start_stop (void)
{
gboolean success = FALSE;
GInetAddress *iaddr;
GSocketAddress *saddr, *listening_addr;
GSocketService *service;
GError *error = NULL;
GSocketClient *client;
iaddr = g_inet_address_new_loopback (G_SOCKET_FAMILY_IPV4);
saddr = g_inet_socket_address_new (iaddr, 0);
g_object_unref (iaddr);
/* instanciate with g_object_new so we can pass active = false */
service = g_object_new (G_TYPE_SOCKET_SERVICE, "active", FALSE, NULL);
g_assert_false (g_socket_service_is_active (service));
g_signal_connect (service, "notify::active", G_CALLBACK (active_notify_cb), &success);
g_socket_listener_add_address (G_SOCKET_LISTENER (service),
saddr,
G_SOCKET_TYPE_STREAM,
G_SOCKET_PROTOCOL_TCP,
NULL,
&listening_addr,
&error);
g_assert_no_error (error);
g_object_unref (saddr);
client = g_socket_client_new ();
g_socket_client_connect_async (client,
G_SOCKET_CONNECTABLE (listening_addr),
NULL,
connected_cb, service);
g_object_unref (client);
g_object_unref (listening_addr);
g_socket_service_start (service);
g_assert_true (g_socket_service_is_active (service));
do
g_main_context_iteration (NULL, TRUE);
while (!success);
g_object_unref (service);
}
int
main (int argc,
char *argv[])
{
g_test_init (&argc, &argv, NULL);
g_test_bug_base ("http://bugzilla.gnome.org/");
g_test_add_func ("/socket-service/start-stop", test_start_stop);
return g_test_run();
}