mirror of
https://github.com/servalproject/serval-dna.git
synced 2025-01-01 10:56:46 +00:00
177 lines
6.4 KiB
C
177 lines
6.4 KiB
C
/*
|
|
Serval Distributed Numbering Architecture (DNA)
|
|
Copyright (C) 2010 Paul Gardner-Stephen
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program 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 General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
/*
|
|
This module is responsible for managing the known list of DID:SID:name mappings,
|
|
and any signatures that vouch for those mappings. Similarly, we wish to be able
|
|
to advertise our mapping and any signatures that vouch for us.
|
|
|
|
We have the choice of using the rhizome database, some other persistent storage
|
|
or an in-memory structure. It would be nice to have a fall-back for when we don't
|
|
have the means to run sqlite. A small in-memory cache will likely be used,
|
|
regardless. This initial implementation assumes that we have sqlite and the rhizome
|
|
database instance available.
|
|
|
|
So what data do we actually need to tell whether we have verified an identity or not?
|
|
We need a DID:SID[:name] tuple.
|
|
We also need the scope for us to mark a given tuple as verified by ourselves, e.g.,
|
|
using a Diffie-Hellman type exchange between handsets.
|
|
Finally, we want the ability to store any signatures that vouch for the identity of
|
|
a given party.
|
|
*/
|
|
|
|
#include "serval.h"
|
|
#include "rhizome.h"
|
|
|
|
/*
|
|
Look up the verification data for the first matching DID, NAME and/or SID.
|
|
(SIDs really should be unique of course).
|
|
If more than one constraint are provided, then the search results are constrained
|
|
by all provided values.
|
|
|
|
Subsequent results can be found by calling dnacache_lookup_next();
|
|
|
|
The operation of this call is greatly simplified by the single-threaded operation
|
|
of DNA, meaning that we don't have to be reentrant, so we can retain a database
|
|
cursor and generally make life simple.
|
|
|
|
So what kind of results do we need to actually return?
|
|
I guess the initial need is for a simple VERIFIED/NOTVERIFIED result, and
|
|
perhaps make the list of certifying identities available. Again the
|
|
single-threaded nature of DNA makes this much simpler than a reentrant version
|
|
would need to be.
|
|
|
|
Database schema is:
|
|
|
|
"CREATE TABLE IF NOT EXISTS VERIFICATIONS(sid text not null, did text, name text,starttime integer, endtime integer,signature blob);"
|
|
|
|
*/
|
|
|
|
/* Cache of verifications.
|
|
This really doesn't need to be too large. */
|
|
#define DNA_IDENTITY_CACHE_SIZE 16
|
|
int dna_identity_cache_initialisedP=0;
|
|
dna_identity_status dna_identity_cache[DNA_IDENTITY_CACHE_SIZE];
|
|
|
|
int dnacache_lookup_next_slot_to_examine=0;
|
|
char *dnacache_lookup_did=NULL;
|
|
char *dnacache_lookup_name=NULL;
|
|
char *dnacache_lookup_sid=NULL;
|
|
int dnacache_lookup_complete=1;
|
|
int dnacache_lookup_sidfound=0;
|
|
|
|
|
|
dna_identity_status *dnacache_lookup(char *did,char *name,char *sid)
|
|
{
|
|
/* Mark all slots as unused initially */
|
|
if (!dna_identity_cache_initialisedP) {
|
|
int i;
|
|
for(i=0;i<DNA_IDENTITY_CACHE_SIZE;i++)
|
|
dna_identity_cache[i].initialisedP=0;
|
|
dna_identity_cache_initialisedP=1;
|
|
}
|
|
|
|
/* Now prepare the query or see if the identity is already in the cache.
|
|
The cache can be used without database query if SID is specified, since
|
|
that guarantees that only one entry exists.
|
|
|
|
Also, if DID/NAME is specified without SID, and the cache entry indicates
|
|
that there are no duplicates for the specified value in the database, then
|
|
the database query can be eshewed, provided that the entry for the SID
|
|
(whether positive or negative) has been pulled into the cache.
|
|
|
|
Otherwise, the database must be queried.
|
|
|
|
Negative results are also cached so that database queries can be avoided for
|
|
repeated queries to identies that we do not know about.
|
|
*/
|
|
|
|
/* Here we just prepare the lookup, and then we do the real work in
|
|
dnacache_lookup_next(), which does the heavy lifting */
|
|
|
|
dnacache_lookup_next_slot_to_examine=0;
|
|
dnacache_lookup_did=did;
|
|
dnacache_lookup_name=name;
|
|
dnacache_lookup_sid=sid;
|
|
dnacache_lookup_complete=0;
|
|
dnacache_lookup_sidfound=0;
|
|
|
|
return dnacache_lookup_next();
|
|
}
|
|
|
|
dna_identity_status *dnacache_lookup_next()
|
|
{
|
|
if (dnacache_lookup_complete) return NULL;
|
|
|
|
/* Look in slots for matches */
|
|
while (dnacache_lookup_next_slot_to_examine<DNA_IDENTITY_CACHE_SIZE)
|
|
{
|
|
dna_identity_status *slot=
|
|
&dna_identity_cache[dnacache_lookup_next_slot_to_examine++];
|
|
|
|
/* Perform the various tests that reject this slot if it doesn't match */
|
|
if (!slot->initialisedP) continue;
|
|
if (dnacache_lookup_sid&&(strcasecmp(dnacache_lookup_sid,slot->sid)))
|
|
continue;
|
|
else dnacache_lookup_sidfound=1;
|
|
if (dnacache_lookup_did&&(strcasecmp(dnacache_lookup_did,slot->did)))
|
|
continue;
|
|
if (dnacache_lookup_name&&(strcasecmp(dnacache_lookup_name,slot->name)))
|
|
continue;
|
|
|
|
if (dnacache_lookup_sid&&dnacache_lookup_sidfound)
|
|
/* SIDs are unique, a SID was specified, and a matching record was found,
|
|
therefore no database query is required. */
|
|
dnacache_lookup_complete=1;
|
|
if (slot->uniqueDidAndName&&dnacache_lookup_did&&dnacache_lookup_name)
|
|
/* If the entry is known to be the only such DID/name combination,
|
|
then we need look no further */
|
|
dnacache_lookup_complete=1;
|
|
|
|
/* Well, we passed all the slots, so it must be us */
|
|
return slot;
|
|
}
|
|
|
|
WHY("Do database lookup");
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
Record/update the verification status of a DID/SID combination.
|
|
|
|
If signature is not null, then add this signature to the list of verifications
|
|
for the specified DID/SID/NAME combination. NAME may be omitted.
|
|
|
|
If revokeVerificationP is non-zero, then all existing verifications for the
|
|
specified identity will be revoked.
|
|
*/
|
|
int dnacache_update_verification(char *did,char *sid,char *name,
|
|
char *signature,int revokeVerificationP)
|
|
{
|
|
return WHY("Not implemented");
|
|
}
|
|
|
|
/* Sign a verification record ourselves for the specified identity, by creating
|
|
and storing signature and adding record to the cache. */
|
|
int dnacache_vouch_for_identity(char *did,char *sid,char *name)
|
|
{
|
|
return WHY("Not implemented");
|
|
}
|
|
|