/* * Copyright (c) 2008 Kungliga Tekniska Högskolan * (Royal Institute of Technology, Stockholm, Sweden). * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * 3. Neither the name of the Institute nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include "krb5_locl.h" #ifdef HAVE_SCC #include typedef struct krb5_scache { char *name; char *file; char *sub; sqlite3 *db; sqlite_uint64 cid; sqlite3_stmt *icred; sqlite3_stmt *dcred; sqlite3_stmt *iprincipal; sqlite3_stmt *icache; sqlite3_stmt *ucachen; sqlite3_stmt *ucachep; sqlite3_stmt *dcache; sqlite3_stmt *scache; sqlite3_stmt *scache_name; sqlite3_stmt *umaster; } krb5_scache; #define SCACHE(X) ((krb5_scache *)(X)->data.data) /* * Because we can't control what permissions SQLite3 (if not in-tree) will use, * and we're a library and can't set the umask. We can't even determine the * current umask in a thread-safe way (not easily), and we can't tell if some * other thread might change it. So what we'll do is put the SQLite3-based * ccache file in its own directory so we can create that directory with * mkdir(2) and the correct permissions. */ #define SCACHE_DEF_NAME "Default-cache" #define KRB5_SCACHE_DIR "%{TEMP}/krb5scc_%{uid}" #define KRB5_SCACHE_DB KRB5_SCACHE_DIR "scc" #define KRB5_SCACHE_NAME "SCC:" KRB5_SCACHE_DB ":" SCACHE_DEF_NAME #define SCACHE_INVALID_CID ((sqlite_uint64)-1) /* * */ #define SQL_CMASTER "" \ "CREATE TABLE master (" \ "oid INTEGER PRIMARY KEY," \ "version INTEGER NOT NULL," \ "defaultcache TEXT NOT NULL" \ ")" #define SQL_SETUP_MASTER \ "INSERT INTO master (version,defaultcache) VALUES(2, \"" SCACHE_DEF_NAME "\")" #define SQL_UMASTER "UPDATE master SET defaultcache=? WHERE version=2" #define SQL_CCACHE "" \ "CREATE TABLE caches (" \ "oid INTEGER PRIMARY KEY," \ "principal TEXT," \ "name TEXT NOT NULL" \ ")" #define SQL_TCACHE "" \ "CREATE TRIGGER CacheDropCreds AFTER DELETE ON caches " \ "FOR EACH ROW BEGIN " \ "DELETE FROM credentials WHERE cid=old.oid;" \ "END" #define SQL_ICACHE "INSERT INTO caches (name) VALUES(?)" #define SQL_UCACHE_NAME "UPDATE caches SET name=? WHERE OID=?" #define SQL_UCACHE_PRINCIPAL "UPDATE caches SET principal=? WHERE OID=?" #define SQL_DCACHE "DELETE FROM caches WHERE OID=?" #define SQL_SCACHE "SELECT principal,name FROM caches WHERE OID=?" #define SQL_SCACHE_NAME "SELECT oid FROM caches WHERE NAME=? OR " \ "(PRINCIPAL IS NOT NULL AND PRINCIPAL=?)" #define SQL_CCREDS "" \ "CREATE TABLE credentials (" \ "oid INTEGER PRIMARY KEY," \ "cid INTEGER NOT NULL," \ "kvno INTEGER NOT NULL," \ "etype INTEGER NOT NULL," \ "created_at INTEGER NOT NULL," \ "cred BLOB NOT NULL" \ ")" #define SQL_TCRED "" \ "CREATE TRIGGER credDropPrincipal AFTER DELETE ON credentials " \ "FOR EACH ROW BEGIN " \ "DELETE FROM principals WHERE credential_id=old.oid;" \ "END" #define SQL_ICRED "INSERT INTO credentials (cid, kvno, etype, cred, created_at) VALUES (?,?,?,?,?)" #define SQL_DCRED "DELETE FROM credentials WHERE cid=?" #define SQL_CPRINCIPALS "" \ "CREATE TABLE principals (" \ "oid INTEGER PRIMARY KEY," \ "principal TEXT NOT NULL," \ "type INTEGER NOT NULL," \ "credential_id INTEGER NOT NULL" \ ")" #define SQL_IPRINCIPAL "INSERT INTO principals (principal, type, credential_id) VALUES (?,?,?)" /* * sqlite destructors */ static void free_data(void *data) { free(data); } static void free_krb5(void *str) { krb5_xfree(str); } static void scc_free(krb5_scache *s) { if (!s) return; if (s->file) free(s->file); if (s->sub) free(s->sub); if (s->name) free(s->name); if (s->icred) sqlite3_finalize(s->icred); if (s->dcred) sqlite3_finalize(s->dcred); if (s->iprincipal) sqlite3_finalize(s->iprincipal); if (s->icache) sqlite3_finalize(s->icache); if (s->ucachen) sqlite3_finalize(s->ucachen); if (s->ucachep) sqlite3_finalize(s->ucachep); if (s->dcache) sqlite3_finalize(s->dcache); if (s->scache) sqlite3_finalize(s->scache); if (s->scache_name) sqlite3_finalize(s->scache_name); if (s->umaster) sqlite3_finalize(s->umaster); if (s->db) sqlite3_close(s->db); free(s); } #ifdef TRACEME static void trace(void* ptr, const char * str) { printf("SQL: %s\n", str); } #endif static krb5_error_code prepare_stmt(krb5_context context, sqlite3 *db, sqlite3_stmt **stmt, const char *str) { int ret; ret = sqlite3_prepare_v2(db, str, -1, stmt, NULL); if (ret != SQLITE_OK) { krb5_set_error_message(context, ENOENT, N_("Failed to prepare stmt %s: %s", ""), str, sqlite3_errmsg(db)); return ENOENT; } return 0; } static krb5_error_code exec_stmt(krb5_context context, sqlite3 *db, const char *str, krb5_error_code code) { int ret; ret = sqlite3_exec(db, str, NULL, NULL, NULL); if (ret != SQLITE_OK && code) { krb5_set_error_message(context, code, N_("scache execute %s: %s", ""), str, sqlite3_errmsg(db)); return code; } return 0; } /* See block comment at the top of this file */ static krb5_error_code make_dir(krb5_context context, const char *name) { krb5_error_code ret = 0; char *s, *p; /* We really need a dirname() in roken; lib/krb5/fcache.c has one */ if ((s = strdup(name)) == NULL) return krb5_enomem(context); for (p = s + strlen(s); p > s; p--) { #ifdef WIN32 if (*p != '/' && *p != '\\') continue; #else if (*p != '/') continue; #endif *p = '\0'; break; } /* If p == s then DB in current directory -- nothing we can do */ if (p > s && mkdir(s, 0700) == -1) ret = errno; free(s); /* If we created it, we're good, else there's nothing we can do */ if (ret == EEXIST) return 0; if (ret) krb5_set_error_message(context, ret, N_("Error making directory for scache file %s", ""), name); return ret; } static krb5_error_code default_db(krb5_context context, const char *name, sqlite3 **db, char **file) { krb5_error_code ret = 0; char *s = NULL; char *f = NULL; if (file) *file = NULL; if (name == NULL) { if ((name = krb5_cc_default_name(context))) { if (strncmp(name, "SCC:", sizeof("SCC:") - 1) == 0) name += sizeof("SCC:") - 1; } if (name == NULL) { ret = _krb5_expand_default_cc_name(context, KRB5_SCACHE_DB, &s); if (ret) return ret; name = s; } } if (strncmp(name, "SCC:", sizeof("SCC:") - 1) == 0) name += sizeof("SCC:") - 1; if ((f = strdup(name)) == NULL) { free(s); return krb5_enomem(context); } free(s); /* Strip off any residue from default name */ #ifdef WIN32 if (f[0] && f[1] == ':' && (s = strrchr(f, ':')) != &f[1]) *s = '\0'; #else if ((s = strrchr(f, ':'))) *s = '\0'; #endif ret = make_dir(context, f); if (ret == 0) { int sret; sret = sqlite3_open_v2(f, db, SQLITE_OPEN_READWRITE, NULL); if (sret != SQLITE_OK) { if (*db) { krb5_set_error_message(context, ENOENT, N_("Error opening scache file %s: %s (%d)", ""), f, sqlite3_errmsg(*db), sret); sqlite3_close(*db); *db = NULL; } else krb5_set_error_message(context, ENOENT, N_("Error opening scache file %s: %s (%d)", ""), f, sqlite3_errstr(sret), sret); free(f); return ENOENT; } } #ifndef WIN32 /* * Just in case we're using an out-of-tree SQLite3. See block comment at * the top of this file, near KRB5_SCACHE_DIR's definition. */ (void) chmod(f, 0600); #endif if (file) *file = f; else free(f); #ifdef TRACEME sqlite3_trace(*db, trace, NULL); #endif return ret; } static krb5_error_code get_def_name(krb5_context context, char *filein, char **str, char **file) { krb5_error_code ret; sqlite3_stmt *stmt; const char *name; sqlite3 *db; ret = default_db(context, filein, &db, file); if (ret) return ret; ret = prepare_stmt(context, db, &stmt, "SELECT defaultcache FROM master"); if (ret) { sqlite3_close(db); return ret; } ret = sqlite3_step(stmt); if (ret != SQLITE_ROW) goto out; if (sqlite3_column_type(stmt, 0) != SQLITE_TEXT) goto out; name = (const char *)sqlite3_column_text(stmt, 0); if (name == NULL) goto out; *str = strdup(name); if (*str == NULL) goto out; sqlite3_finalize(stmt); sqlite3_close(db); return 0; out: sqlite3_finalize(stmt); sqlite3_close(db); krb5_clear_error_message(context); return ENOENT; } static krb5_scache * KRB5_CALLCONV scc_alloc(krb5_context context, const char *name, const char *sub, int new_unique) { krb5_error_code ret = 0; krb5_scache *s; char *freeme = NULL; char *subsidiary; ALLOC(s, 1); if(s == NULL) return NULL; s->cid = SCACHE_INVALID_CID; if (name && *name && sub && *sub) { if ((s->sub = strdup(sub)) == NULL || (s->file = strdup(name)) == NULL) { free(s->file); free(s); (void) krb5_enomem(context); return NULL; } } else { s->sub = NULL; s->file = NULL; s->name = NULL; if (name == NULL) name = krb5_cc_default_name(context); if (name == NULL) { ret = _krb5_expand_default_cc_name(context, KRB5_SCACHE_DB, &freeme); if (ret) { free(s); return NULL; } name = freeme; } if (strncmp(name, "SCC:", sizeof("SCC:") - 1) == 0) name += sizeof("SCC:") - 1; if ((s->file = strdup(name)) == NULL) { ret = krb5_enomem(context); goto out; } if ((subsidiary = strrchr(s->file, ':'))) { #ifdef WIN32 if (subsidiary == s->file + 1) subsidiary = NULL; else #endif *(subsidiary++) = '\0'; } if (new_unique) { ret = asprintf(&s->sub, "unique-%p", s) < 0 || s->sub == NULL ? krb5_enomem(context) : 0; } else if (subsidiary == NULL || *subsidiary == '\0') { ret = get_def_name(context, s->file, &s->sub, NULL); if (ret) { if ((s->sub = strdup(SCACHE_DEF_NAME)) == NULL) ret = krb5_enomem(context); else ret = 0; } } else if ((s->sub = strdup(subsidiary)) == NULL) { ret = krb5_enomem(context); } } if (ret == 0 && s->file && s->sub && (asprintf(&s->name, "%s:%s", s->file, s->sub) < 0 || s->name == NULL)) ret = krb5_enomem(context); out: if (ret || s->file == NULL || s->sub == NULL || s->name == NULL) { scc_free(s); s = NULL; } free(freeme); return s; } static krb5_error_code open_database(krb5_context context, krb5_scache *s, int flags) { krb5_error_code ret; struct stat st; int sret; if (!(flags & SQLITE_OPEN_CREATE) && stat(s->file, &st) == 0 && st.st_size == 0) return ENOENT; ret = make_dir(context, s->file); if (ret) return ret; sret = sqlite3_open_v2(s->file, &s->db, SQLITE_OPEN_READWRITE|flags, NULL); if (sret != SQLITE_OK) { if (s->db) { krb5_set_error_message(context, ENOENT, N_("Error opening scache file %s: %s (%d)", ""), s->file, sqlite3_errmsg(s->db), sret); sqlite3_close(s->db); s->db = NULL; } else krb5_set_error_message(context, ENOENT, N_("Error opening scache file %s: %s (%d)", ""), s->file, sqlite3_errstr(sret), sret); return ENOENT; } return 0; } static krb5_error_code create_cache(krb5_context context, krb5_scache *s) { int ret; sqlite3_bind_text(s->icache, 1, s->sub, -1, NULL); do { ret = sqlite3_step(s->icache); } while (ret == SQLITE_ROW); if (ret != SQLITE_DONE) { krb5_set_error_message(context, KRB5_CC_IO, N_("Failed to add scache: %d", ""), ret); return KRB5_CC_IO; } sqlite3_reset(s->icache); s->cid = sqlite3_last_insert_rowid(s->db); return 0; } static krb5_error_code make_database(krb5_context context, krb5_scache *s) { int created_file = 0; int ret; if (s->db) return 0; ret = open_database(context, s, 0); if (ret) { ret = open_database(context, s, SQLITE_OPEN_CREATE); if (ret) goto out; created_file = 1; ret = exec_stmt(context, s->db, SQL_CMASTER, KRB5_CC_IO); if (ret) goto out; ret = exec_stmt(context, s->db, SQL_CCACHE, KRB5_CC_IO); if (ret) goto out; ret = exec_stmt(context, s->db, SQL_CCREDS, KRB5_CC_IO); if (ret) goto out; ret = exec_stmt(context, s->db, SQL_CPRINCIPALS, KRB5_CC_IO); if (ret) goto out; ret = exec_stmt(context, s->db, SQL_SETUP_MASTER, KRB5_CC_IO); if (ret) goto out; ret = exec_stmt(context, s->db, SQL_TCACHE, KRB5_CC_IO); if (ret) goto out; ret = exec_stmt(context, s->db, SQL_TCRED, KRB5_CC_IO); if (ret) goto out; } #ifdef TRACEME sqlite3_trace(s->db, trace, NULL); #endif ret = prepare_stmt(context, s->db, &s->icred, SQL_ICRED); if (ret) goto out; ret = prepare_stmt(context, s->db, &s->dcred, SQL_DCRED); if (ret) goto out; ret = prepare_stmt(context, s->db, &s->iprincipal, SQL_IPRINCIPAL); if (ret) goto out; ret = prepare_stmt(context, s->db, &s->icache, SQL_ICACHE); if (ret) goto out; ret = prepare_stmt(context, s->db, &s->ucachen, SQL_UCACHE_NAME); if (ret) goto out; ret = prepare_stmt(context, s->db, &s->ucachep, SQL_UCACHE_PRINCIPAL); if (ret) goto out; ret = prepare_stmt(context, s->db, &s->dcache, SQL_DCACHE); if (ret) goto out; ret = prepare_stmt(context, s->db, &s->scache, SQL_SCACHE); if (ret) goto out; ret = prepare_stmt(context, s->db, &s->scache_name, SQL_SCACHE_NAME); if (ret) goto out; ret = prepare_stmt(context, s->db, &s->umaster, SQL_UMASTER); if (ret) goto out; #ifndef WIN32 /* * Just in case we're using an out-of-tree SQLite3. See block comment at * the top of this file, near KRB5_SCACHE_DIR's definition. */ (void) chmod(s->file, 0600); #endif return 0; out: if (s->db) sqlite3_close(s->db); if (created_file) unlink(s->file); return ret; } static krb5_error_code bind_principal(krb5_context context, sqlite3 *db, sqlite3_stmt *stmt, int col, krb5_const_principal principal) { krb5_error_code ret; char *str; ret = krb5_unparse_name(context, principal, &str); if (ret) return ret; ret = sqlite3_bind_text(stmt, col, str, -1, free_krb5); if (ret != SQLITE_OK) { krb5_xfree(str); krb5_set_error_message(context, ENOMEM, N_("scache bind principal: %s", ""), sqlite3_errmsg(db)); return ENOMEM; } return 0; } /* * */ static krb5_error_code KRB5_CALLCONV scc_get_name_2(krb5_context context, krb5_ccache id, const char **name, const char **file, const char **sub) { if (name) *name = SCACHE(id)->name; if (file) *file = SCACHE(id)->file; if (sub) *sub = SCACHE(id)->sub; return 0; } static krb5_error_code KRB5_CALLCONV scc_resolve_2(krb5_context context, krb5_ccache *id, const char *res, const char *sub) { krb5_error_code ret; krb5_scache *s; s = scc_alloc(context, res, sub, 0); if (s == NULL) { krb5_set_error_message(context, KRB5_CC_NOMEM, N_("malloc: out of memory", "")); return KRB5_CC_NOMEM; } ret = make_database(context, s); if (ret) { scc_free(s); return ret; } ret = sqlite3_bind_text(s->scache_name, 1, s->sub, -1, NULL); if (ret != SQLITE_OK) { krb5_set_error_message(context, ENOMEM, "bind principal: %s", sqlite3_errmsg(s->db)); scc_free(s); return ENOMEM; } if (sqlite3_step(s->scache_name) == SQLITE_ROW) { if (sqlite3_column_type(s->scache_name, 0) != SQLITE_INTEGER) { sqlite3_reset(s->scache_name); krb5_set_error_message(context, KRB5_CC_END, N_("Cache name of wrong type " "for scache %s", ""), s->name); scc_free(s); return KRB5_CC_END; } s->cid = sqlite3_column_int(s->scache_name, 0); } else { s->cid = SCACHE_INVALID_CID; } sqlite3_reset(s->scache_name); (*id)->data.data = s; (*id)->data.length = sizeof(*s); return 0; } static krb5_error_code KRB5_CALLCONV scc_gen_new(krb5_context context, krb5_ccache *id) { krb5_scache *s; s = scc_alloc(context, NULL, NULL, 1); if (s == NULL) { krb5_set_error_message(context, KRB5_CC_NOMEM, N_("malloc: out of memory", "")); return KRB5_CC_NOMEM; } (*id)->data.data = s; (*id)->data.length = sizeof(*s); return 0; } static krb5_error_code KRB5_CALLCONV scc_initialize(krb5_context context, krb5_ccache id, krb5_principal principal) { krb5_scache *s = SCACHE(id); krb5_error_code ret; ret = make_database(context, s); if (ret) return ret; ret = exec_stmt(context, s->db, "BEGIN IMMEDIATE TRANSACTION", KRB5_CC_IO); if (ret) return ret; if (s->cid == SCACHE_INVALID_CID) { ret = create_cache(context, s); if (ret) goto rollback; } else { sqlite3_bind_int(s->dcred, 1, s->cid); do { ret = sqlite3_step(s->dcred); } while (ret == SQLITE_ROW); sqlite3_reset(s->dcred); if (ret != SQLITE_DONE) { ret = KRB5_CC_IO; krb5_set_error_message(context, ret, N_("Failed to delete old " "credentials: %s", ""), sqlite3_errmsg(s->db)); goto rollback; } } ret = bind_principal(context, s->db, s->ucachep, 1, principal); if (ret) goto rollback; sqlite3_bind_int(s->ucachep, 2, s->cid); do { ret = sqlite3_step(s->ucachep); } while (ret == SQLITE_ROW); sqlite3_reset(s->ucachep); if (ret != SQLITE_DONE) { ret = KRB5_CC_IO; krb5_set_error_message(context, ret, N_("Failed to bind principal to cache %s", ""), sqlite3_errmsg(s->db)); goto rollback; } ret = exec_stmt(context, s->db, "COMMIT", KRB5_CC_IO); if (ret) return ret; return 0; rollback: exec_stmt(context, s->db, "ROLLBACK", 0); return ret; } static krb5_error_code KRB5_CALLCONV scc_close(krb5_context context, krb5_ccache id) { scc_free(SCACHE(id)); return 0; } static krb5_error_code KRB5_CALLCONV scc_destroy(krb5_context context, krb5_ccache id) { krb5_scache *s = SCACHE(id); int ret; if (s->cid == SCACHE_INVALID_CID) return 0; sqlite3_bind_int(s->dcache, 1, s->cid); do { ret = sqlite3_step(s->dcache); } while (ret == SQLITE_ROW); sqlite3_reset(s->dcache); if (ret != SQLITE_DONE) { krb5_set_error_message(context, KRB5_CC_IO, N_("Failed to destroy cache %s: %s", ""), s->name, sqlite3_errmsg(s->db)); return KRB5_CC_IO; } return 0; } static krb5_error_code encode_creds(krb5_context context, krb5_creds *creds, krb5_data *data) { krb5_error_code ret; krb5_storage *sp; krb5_data_zero(data); sp = krb5_storage_emem(); if (sp == NULL) return krb5_enomem(context); ret = krb5_store_creds(sp, creds); if (ret) { krb5_set_error_message(context, ret, N_("Failed to store credential in scache", "")); krb5_storage_free(sp); return ret; } ret = krb5_storage_to_data(sp, data); krb5_storage_free(sp); if (ret) krb5_set_error_message(context, ret, N_("Failed to encode credential in scache", "")); return ret; } static krb5_error_code decode_creds(krb5_context context, const void *data, size_t length, krb5_creds *creds) { krb5_error_code ret; krb5_storage *sp; sp = krb5_storage_from_readonly_mem(data, length); if (sp == NULL) return krb5_enomem(context); ret = krb5_ret_creds(sp, creds); krb5_storage_free(sp); if (ret) { krb5_set_error_message(context, ret, N_("Failed to read credential in scache", "")); return ret; } return 0; } static krb5_error_code KRB5_CALLCONV scc_store_cred(krb5_context context, krb5_ccache id, krb5_creds *creds) { sqlite_uint64 credid; krb5_scache *s = SCACHE(id); krb5_error_code ret; krb5_data data; ret = make_database(context, s); if (ret) return ret; ret = encode_creds(context, creds, &data); if (ret) return ret; sqlite3_bind_int(s->icred, 1, s->cid); { krb5_enctype etype = 0; int kvno = 0; Ticket t; size_t len; ret = decode_Ticket(creds->ticket.data, creds->ticket.length, &t, &len); if (ret == 0) { if(t.enc_part.kvno) kvno = *t.enc_part.kvno; etype = t.enc_part.etype; free_Ticket(&t); } sqlite3_bind_int(s->icred, 2, kvno); sqlite3_bind_int(s->icred, 3, etype); } sqlite3_bind_blob(s->icred, 4, data.data, data.length, free_data); sqlite3_bind_int(s->icred, 5, time(NULL)); ret = exec_stmt(context, s->db, "BEGIN IMMEDIATE TRANSACTION", KRB5_CC_IO); if (ret) return ret; do { ret = sqlite3_step(s->icred); } while (ret == SQLITE_ROW); sqlite3_reset(s->icred); if (ret != SQLITE_DONE) { ret = KRB5_CC_IO; krb5_set_error_message(context, ret, N_("Failed to add credential: %s", ""), sqlite3_errmsg(s->db)); goto rollback; } credid = sqlite3_last_insert_rowid(s->db); { bind_principal(context, s->db, s->iprincipal, 1, creds->server); sqlite3_bind_int(s->iprincipal, 2, 1); sqlite3_bind_int(s->iprincipal, 3, credid); do { ret = sqlite3_step(s->iprincipal); } while (ret == SQLITE_ROW); sqlite3_reset(s->iprincipal); if (ret != SQLITE_DONE) { ret = KRB5_CC_IO; krb5_set_error_message(context, ret, N_("Failed to add principal: %s", ""), sqlite3_errmsg(s->db)); goto rollback; } } { bind_principal(context, s->db, s->iprincipal, 1, creds->client); sqlite3_bind_int(s->iprincipal, 2, 0); sqlite3_bind_int(s->iprincipal, 3, credid); do { ret = sqlite3_step(s->iprincipal); } while (ret == SQLITE_ROW); sqlite3_reset(s->iprincipal); if (ret != SQLITE_DONE) { ret = KRB5_CC_IO; krb5_set_error_message(context, ret, N_("Failed to add principal: %s", ""), sqlite3_errmsg(s->db)); goto rollback; } } ret = exec_stmt(context, s->db, "COMMIT", KRB5_CC_IO); if (ret) return ret; return 0; rollback: exec_stmt(context, s->db, "ROLLBACK", 0); return ret; } static krb5_error_code KRB5_CALLCONV scc_get_principal(krb5_context context, krb5_ccache id, krb5_principal *principal) { krb5_scache *s = SCACHE(id); krb5_error_code ret; const char *str; *principal = NULL; ret = make_database(context, s); if (ret) return ret; sqlite3_bind_int(s->scache, 1, s->cid); if (sqlite3_step(s->scache) != SQLITE_ROW) { sqlite3_reset(s->scache); krb5_set_error_message(context, KRB5_CC_END, N_("No principal for cache SCC:%s", ""), s->name); return KRB5_CC_END; } if (sqlite3_column_type(s->scache, 0) != SQLITE_TEXT) { sqlite3_reset(s->scache); krb5_set_error_message(context, KRB5_CC_END, N_("Principal data of wrong type " "for SCC:%s", ""), s->name); return KRB5_CC_END; } str = (const char *)sqlite3_column_text(s->scache, 0); if (str == NULL) { sqlite3_reset(s->scache); krb5_set_error_message(context, KRB5_CC_END, N_("Principal not set for SCC:%s", ""), s->name); return KRB5_CC_END; } ret = krb5_parse_name(context, str, principal); sqlite3_reset(s->scache); return ret; } struct cred_ctx { char *drop; sqlite3_stmt *stmt; sqlite3_stmt *credstmt; }; static krb5_error_code KRB5_CALLCONV scc_get_first (krb5_context context, krb5_ccache id, krb5_cc_cursor *cursor) { krb5_scache *s = SCACHE(id); krb5_error_code ret; struct cred_ctx *ctx; char *str = NULL, *name = NULL; *cursor = NULL; ctx = calloc(1, sizeof(*ctx)); if (ctx == NULL) return krb5_enomem(context); ret = make_database(context, s); if (ret) { free(ctx); return ret; } if (s->cid == SCACHE_INVALID_CID) { krb5_set_error_message(context, KRB5_CC_END, N_("Iterating a invalid scache %s", ""), s->name); free(ctx); return KRB5_CC_END; } ret = asprintf(&name, "credIteration%pPid%d", ctx, (int)getpid()); if (ret < 0 || name == NULL) { free(ctx); return krb5_enomem(context); } ret = asprintf(&ctx->drop, "DROP TABLE %s", name); if (ret < 0 || ctx->drop == NULL) { free(name); free(ctx); return krb5_enomem(context); } ret = asprintf(&str, "CREATE TEMPORARY TABLE %s " "AS SELECT oid,created_at FROM credentials WHERE cid = %lu", name, (unsigned long)s->cid); if (ret < 0 || str == NULL) { free(ctx->drop); free(name); free(ctx); return krb5_enomem(context); } ret = exec_stmt(context, s->db, str, KRB5_CC_IO); free(str); str = NULL; if (ret) { free(ctx->drop); free(name); free(ctx); return ret; } ret = asprintf(&str, "SELECT oid FROM %s ORDER BY created_at", name); if (ret < 0 || str == NULL) { exec_stmt(context, s->db, ctx->drop, 0); free(ctx->drop); free(name); free(ctx); return ret; } ret = prepare_stmt(context, s->db, &ctx->stmt, str); free(str); str = NULL; free(name); if (ret) { exec_stmt(context, s->db, ctx->drop, 0); free(ctx->drop); free(ctx); return ret; } ret = prepare_stmt(context, s->db, &ctx->credstmt, "SELECT cred FROM credentials WHERE oid = ?"); if (ret) { sqlite3_finalize(ctx->stmt); exec_stmt(context, s->db, ctx->drop, 0); free(ctx->drop); free(ctx); return ret; } *cursor = ctx; return 0; } static krb5_error_code KRB5_CALLCONV scc_get_next (krb5_context context, krb5_ccache id, krb5_cc_cursor *cursor, krb5_creds *creds) { struct cred_ctx *ctx = *cursor; krb5_scache *s = SCACHE(id); krb5_error_code ret; sqlite_uint64 oid; const void *data = NULL; size_t len = 0; next: ret = sqlite3_step(ctx->stmt); if (ret == SQLITE_DONE) { krb5_clear_error_message(context); return KRB5_CC_END; } else if (ret != SQLITE_ROW) { krb5_set_error_message(context, KRB5_CC_IO, N_("scache Database failed: %s", ""), sqlite3_errmsg(s->db)); return KRB5_CC_IO; } oid = sqlite3_column_int64(ctx->stmt, 0); /* read cred from credentials table */ sqlite3_bind_int(ctx->credstmt, 1, oid); ret = sqlite3_step(ctx->credstmt); if (ret != SQLITE_ROW) { sqlite3_reset(ctx->credstmt); goto next; } if (sqlite3_column_type(ctx->credstmt, 0) != SQLITE_BLOB) { krb5_set_error_message(context, KRB5_CC_END, N_("credential of wrong type for SCC:%s", ""), s->name); sqlite3_reset(ctx->credstmt); return KRB5_CC_END; } data = sqlite3_column_blob(ctx->credstmt, 0); len = sqlite3_column_bytes(ctx->credstmt, 0); ret = decode_creds(context, data, len, creds); sqlite3_reset(ctx->credstmt); return ret; } static krb5_error_code KRB5_CALLCONV scc_end_get (krb5_context context, krb5_ccache id, krb5_cc_cursor *cursor) { struct cred_ctx *ctx = *cursor; krb5_scache *s = SCACHE(id); sqlite3_finalize(ctx->stmt); sqlite3_finalize(ctx->credstmt); exec_stmt(context, s->db, ctx->drop, 0); free(ctx->drop); free(ctx); return 0; } static krb5_error_code KRB5_CALLCONV scc_remove_cred(krb5_context context, krb5_ccache id, krb5_flags which, krb5_creds *mcreds) { krb5_scache *s = SCACHE(id); krb5_error_code ret; sqlite3_stmt *stmt; sqlite_uint64 credid = 0; const void *data = NULL; size_t len = 0; ret = make_database(context, s); if (ret) return ret; ret = prepare_stmt(context, s->db, &stmt, "SELECT cred,oid FROM credentials " "WHERE cid = ?"); if (ret) return ret; sqlite3_bind_int(stmt, 1, s->cid); /* find credential... */ while (1) { krb5_creds creds; ret = sqlite3_step(stmt); if (ret == SQLITE_DONE) { ret = 0; break; } else if (ret != SQLITE_ROW) { ret = KRB5_CC_IO; krb5_set_error_message(context, ret, N_("scache Database failed: %s", ""), sqlite3_errmsg(s->db)); break; } if (sqlite3_column_type(stmt, 0) != SQLITE_BLOB) { ret = KRB5_CC_END; krb5_set_error_message(context, ret, N_("Credential of wrong type " "for SCC:%s", ""), s->name); break; } data = sqlite3_column_blob(stmt, 0); len = sqlite3_column_bytes(stmt, 0); ret = decode_creds(context, data, len, &creds); if (ret) break; ret = krb5_compare_creds(context, which, mcreds, &creds); krb5_free_cred_contents(context, &creds); if (ret) { credid = sqlite3_column_int64(stmt, 1); ret = 0; break; } } sqlite3_finalize(stmt); ret = prepare_stmt(context, s->db, &stmt, "DELETE FROM credentials WHERE oid=?"); if (ret) return ret; sqlite3_bind_int(stmt, 1, credid); do { ret = sqlite3_step(stmt); } while (ret == SQLITE_ROW); sqlite3_finalize(stmt); if (ret != SQLITE_DONE) { ret = KRB5_CC_IO; krb5_set_error_message(context, ret, N_("failed to delete scache credental", "")); } else ret = 0; return ret; } static krb5_error_code KRB5_CALLCONV scc_set_flags(krb5_context context, krb5_ccache id, krb5_flags flags) { return 0; /* XXX */ } struct cache_iter { char *drop; char *file; sqlite3 *db; sqlite3_stmt *stmt; }; static krb5_error_code KRB5_CALLCONV scc_get_cache_first(krb5_context context, krb5_cc_cursor *cursor) { struct cache_iter *ctx; krb5_error_code ret; char *name = NULL, *str = NULL; *cursor = NULL; ctx = calloc(1, sizeof(*ctx)); if (ctx == NULL) return krb5_enomem(context); ret = default_db(context, NULL, &ctx->db, &ctx->file); if (ret) { free(ctx); return ret; } ret = asprintf(&name, "cacheIteration%pPid%d", ctx, (int)getpid()); if (ret < 0 || name == NULL) { sqlite3_close(ctx->db); free(ctx); return krb5_enomem(context); } ret = asprintf(&ctx->drop, "DROP TABLE %s", name); if (ret < 0 || ctx->drop == NULL) { sqlite3_close(ctx->db); free(name); free(ctx); return krb5_enomem(context); } ret = asprintf(&str, "CREATE TEMPORARY TABLE %s AS SELECT name FROM caches", name); if (ret < 0 || str == NULL) { sqlite3_close(ctx->db); free(name); free(ctx->drop); free(ctx); return krb5_enomem(context); } ret = exec_stmt(context, ctx->db, str, KRB5_CC_IO); free(str); str = NULL; if (ret) { sqlite3_close(ctx->db); free(name); free(ctx->drop); free(ctx); return ret; } ret = asprintf(&str, "SELECT name FROM %s", name); if (ret < 0 || str == NULL) { exec_stmt(context, ctx->db, ctx->drop, 0); sqlite3_close(ctx->db); free(name); free(ctx->drop); free(ctx); return krb5_enomem(context); } free(name); ret = prepare_stmt(context, ctx->db, &ctx->stmt, str); free(str); if (ret) { exec_stmt(context, ctx->db, ctx->drop, 0); sqlite3_close(ctx->db); free(ctx->drop); free(ctx); return ret; } *cursor = ctx; return 0; } static krb5_error_code KRB5_CALLCONV scc_get_cache_next(krb5_context context, krb5_cc_cursor cursor, krb5_ccache *id) { struct cache_iter *ctx = cursor; krb5_error_code ret; const char *name; again: ret = sqlite3_step(ctx->stmt); if (ret == SQLITE_DONE) { krb5_clear_error_message(context); return KRB5_CC_END; } else if (ret != SQLITE_ROW) { krb5_set_error_message(context, KRB5_CC_IO, N_("Database failed: %s", ""), sqlite3_errmsg(ctx->db)); return KRB5_CC_IO; } if (sqlite3_column_type(ctx->stmt, 0) != SQLITE_TEXT) goto again; name = (const char *)sqlite3_column_text(ctx->stmt, 0); if (name == NULL) goto again; ret = _krb5_cc_allocate(context, &krb5_scc_ops, id); if (ret == 0) ret = scc_resolve_2(context, id, ctx->file, name); if (ret) { free(*id); *id = NULL; } return ret; } static krb5_error_code KRB5_CALLCONV scc_end_cache_get(krb5_context context, krb5_cc_cursor cursor) { struct cache_iter *ctx = cursor; exec_stmt(context, ctx->db, ctx->drop, 0); sqlite3_finalize(ctx->stmt); sqlite3_close(ctx->db); free(ctx->file); free(ctx->drop); free(ctx); return 0; } static krb5_error_code KRB5_CALLCONV scc_move(krb5_context context, krb5_ccache from, krb5_ccache to) { krb5_scache *sfrom = SCACHE(from); krb5_scache *sto = SCACHE(to); krb5_error_code ret; if (strcmp(sfrom->file, sto->file) != 0) { /* Let upstairs handle the move */ return EXDEV; } ret = make_database(context, sfrom); if (ret) return ret; ret = exec_stmt(context, sfrom->db, "BEGIN IMMEDIATE TRANSACTION", KRB5_CC_IO); if (ret) return ret; if (sto->cid != SCACHE_INVALID_CID) { /* drop old cache entry */ sqlite3_bind_int(sfrom->dcache, 1, sto->cid); do { ret = sqlite3_step(sfrom->dcache); } while (ret == SQLITE_ROW); sqlite3_reset(sfrom->dcache); if (ret != SQLITE_DONE) { krb5_set_error_message(context, KRB5_CC_IO, N_("Failed to delete old cache: %d", ""), (int)ret); goto rollback; } } sqlite3_bind_text(sfrom->ucachen, 1, sto->sub, -1, NULL); sqlite3_bind_int(sfrom->ucachen, 2, sfrom->cid); do { ret = sqlite3_step(sfrom->ucachen); } while (ret == SQLITE_ROW); sqlite3_reset(sfrom->ucachen); if (ret != SQLITE_DONE) { krb5_set_error_message(context, KRB5_CC_IO, N_("Failed to update new cache: %d", ""), (int)ret); goto rollback; } sto->cid = sfrom->cid; ret = exec_stmt(context, sfrom->db, "COMMIT", KRB5_CC_IO); if (ret) return ret; krb5_cc_close(context, from); return 0; rollback: exec_stmt(context, sfrom->db, "ROLLBACK", 0); return KRB5_CC_IO; } static krb5_error_code KRB5_CALLCONV scc_get_default_name(krb5_context context, char **str) { *str = NULL; return _krb5_expand_default_cc_name(context, KRB5_SCACHE_NAME, str); } static krb5_error_code KRB5_CALLCONV scc_set_default(krb5_context context, krb5_ccache id) { krb5_scache *s = SCACHE(id); krb5_error_code ret; if (s->cid == SCACHE_INVALID_CID) { krb5_set_error_message(context, KRB5_CC_IO, N_("Trying to set a invalid cache " "as default %s", ""), s->name); return KRB5_CC_IO; } ret = sqlite3_bind_text(s->umaster, 1, s->sub, -1, NULL); if (ret) { sqlite3_reset(s->umaster); krb5_set_error_message(context, KRB5_CC_IO, N_("Failed to set name of default cache", "")); return KRB5_CC_IO; } do { ret = sqlite3_step(s->umaster); } while (ret == SQLITE_ROW); sqlite3_reset(s->umaster); if (ret != SQLITE_DONE) { krb5_set_error_message(context, KRB5_CC_IO, N_("Failed to update default cache", "")); return KRB5_CC_IO; } return 0; } /** * Variable containing the SCC based credential cache implemention. * * @ingroup krb5_ccache */ KRB5_LIB_VARIABLE const krb5_cc_ops krb5_scc_ops = { KRB5_CC_OPS_VERSION_5, "SCC", NULL, NULL, scc_gen_new, scc_initialize, scc_destroy, scc_close, scc_store_cred, NULL, /* scc_retrieve */ scc_get_principal, scc_get_first, scc_get_next, scc_end_get, scc_remove_cred, scc_set_flags, NULL, scc_get_cache_first, scc_get_cache_next, scc_end_cache_get, scc_move, scc_get_default_name, scc_set_default, NULL, NULL, NULL, scc_get_name_2, scc_resolve_2 }; #endif