<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN"
"http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head><meta http-equiv="content-type" content="text/html; charset=utf-8" />
<title>[12436] CalendarServer/trunk/bin/proxyclean</title>
</head>
<body>

<style type="text/css"><!--
#msg dl.meta { border: 1px #006 solid; background: #369; padding: 6px; color: #fff; }
#msg dl.meta dt { float: left; width: 6em; font-weight: bold; }
#msg dt:after { content:':';}
#msg dl, #msg dt, #msg ul, #msg li, #header, #footer, #logmsg { font-family: verdana,arial,helvetica,sans-serif; font-size: 10pt;  }
#msg dl a { font-weight: bold}
#msg dl a:link    { color:#fc3; }
#msg dl a:active  { color:#ff0; }
#msg dl a:visited { color:#cc6; }
h3 { font-family: verdana,arial,helvetica,sans-serif; font-size: 10pt; font-weight: bold; }
#msg pre { overflow: auto; background: #ffc; border: 1px #fa0 solid; padding: 6px; }
#logmsg { background: #ffc; border: 1px #fa0 solid; padding: 1em 1em 0 1em; }
#logmsg p, #logmsg pre, #logmsg blockquote { margin: 0 0 1em 0; }
#logmsg p, #logmsg li, #logmsg dt, #logmsg dd { line-height: 14pt; }
#logmsg h1, #logmsg h2, #logmsg h3, #logmsg h4, #logmsg h5, #logmsg h6 { margin: .5em 0; }
#logmsg h1:first-child, #logmsg h2:first-child, #logmsg h3:first-child, #logmsg h4:first-child, #logmsg h5:first-child, #logmsg h6:first-child { margin-top: 0; }
#logmsg ul, #logmsg ol { padding: 0; list-style-position: inside; margin: 0 0 0 1em; }
#logmsg ul { text-indent: -1em; padding-left: 1em; }#logmsg ol { text-indent: -1.5em; padding-left: 1.5em; }
#logmsg > ul, #logmsg > ol { margin: 0 0 1em 0; }
#logmsg pre { background: #eee; padding: 1em; }
#logmsg blockquote { border: 1px solid #fa0; border-left-width: 10px; padding: 1em 1em 0 1em; background: white;}
#logmsg dl { margin: 0; }
#logmsg dt { font-weight: bold; }
#logmsg dd { margin: 0; padding: 0 0 0.5em 0; }
#logmsg dd:before { content:'\00bb';}
#logmsg table { border-spacing: 0px; border-collapse: collapse; border-top: 4px solid #fa0; border-bottom: 1px solid #fa0; background: #fff; }
#logmsg table th { text-align: left; font-weight: normal; padding: 0.2em 0.5em; border-top: 1px dotted #fa0; }
#logmsg table td { text-align: right; border-top: 1px dotted #fa0; padding: 0.2em 0.5em; }
#logmsg table thead th { text-align: center; border-bottom: 1px solid #fa0; }
#logmsg table th.Corner { text-align: left; }
#logmsg hr { border: none 0; border-top: 2px dashed #fa0; height: 1px; }
#header, #footer { color: #fff; background: #636; border: 1px #300 solid; padding: 6px; }
#patch { width: 100%; }
#patch h4 {font-family: verdana,arial,helvetica,sans-serif;font-size:10pt;padding:8px;background:#369;color:#fff;margin:0;}
#patch .propset h4, #patch .binary h4 {margin:0;}
#patch pre {padding:0;line-height:1.2em;margin:0;}
#patch .diff {width:100%;background:#eee;padding: 0 0 10px 0;overflow:auto;}
#patch .propset .diff, #patch .binary .diff  {padding:10px 0;}
#patch span {display:block;padding:0 10px;}
#patch .modfile, #patch .addfile, #patch .delfile, #patch .propset, #patch .binary, #patch .copfile {border:1px solid #ccc;margin:10px 0;}
#patch ins {background:#dfd;text-decoration:none;display:block;padding:0 10px;}
#patch del {background:#fdd;text-decoration:none;display:block;padding:0 10px;}
#patch .lines, .info {color:#888;background:#fff;}
--></style>
<div id="msg">
<dl class="meta">
<dt>Revision</dt> <dd><a href="http://trac.calendarserver.org//changeset/12436">12436</a></dd>
<dt>Author</dt> <dd>wsanchez@apple.com</dd>
<dt>Date</dt> <dd>2014-01-23 15:53:21 -0800 (Thu, 23 Jan 2014)</dd>
</dl>

<h3>Log Message</h3>
<pre>Nuke</pre>

<h3>Removed Paths</h3>
<ul>
<li><a href="#CalendarServertrunkbinproxyclean">CalendarServer/trunk/bin/proxyclean</a></li>
</ul>

</div>
<div id="patch">
<h3>Diff</h3>
<a id="CalendarServertrunkbinproxyclean"></a>
<div class="delfile"><h4>Deleted: CalendarServer/trunk/bin/proxyclean (12435 => 12436)</h4>
<pre class="diff"><span>
<span class="info">--- CalendarServer/trunk/bin/proxyclean        2014-01-23 23:40:51 UTC (rev 12435)
+++ CalendarServer/trunk/bin/proxyclean        2014-01-23 23:53:21 UTC (rev 12436)
</span><span class="lines">@@ -1,200 +0,0 @@
</span><del>-#!/usr/bin/env python
-
-##
-# Copyright (c) 2008-2014 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the &quot;License&quot;);
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an &quot;AS IS&quot; BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-import sys
-import commands
-import getopt
-import os
-
-try:
-    import _calendarserver_preamble
-except ImportError:
-    sys.exc_clear()
-
-from plistlib import readPlist
-
-try:
-    import opendirectory
-    import dsattributes
-except ImportError:
-    sys.path.append(&quot;/usr/share/caldavd/lib/python&quot;)
-    import opendirectory
-    import dsattributes
-
-def usage(e=None):
-    if e:
-        print e
-        print &quot;&quot;
-
-    name = os.path.basename(sys.argv[0])
-    print &quot;usage: %s [-c] [-f FILE]&quot; % (name,)
-    print &quot;&quot;
-    print &quot;Tool to remove invalid OD record GUIDs from the proxy DB.&quot;
-    print &quot;&quot;
-    print &quot;options:&quot;
-    print &quot;  -h: print this help&quot;
-    print &quot;  -c: change sqlite DB - without this just report what would happen&quot;
-    print &quot;  -f: caldavd.plist file to read (default: /etc/caldavd/caldavd.plist)&quot;
-
-    if e:
-        sys.exit(64)
-    else:
-        sys.exit(0)
-
-def extractPlistPieces(plistdbpath):
-    
-    plist = readPlist(plistdbpath)
-
-    try:
-        dsnode = plist[&quot;DirectoryService&quot;][&quot;params&quot;][&quot;node&quot;]
-    except KeyError:
-        raise ValueError(&quot;Unable to read DirectoryService/params/node key from plist: %s&quot; % (plistdbpath,))
-    
-    try:
-        dataroot = plist[&quot;DataRoot&quot;]
-    except KeyError:
-        raise ValueError(&quot;Unable to read DataRoot key from plist: %s&quot; % (plistdbpath,))
-    
-    # Find the appropriate sqlite db
-    proxydbpath_data = os.path.join(dataroot, &quot;calendaruserproxy.sqlite&quot;)
-    if not os.path.exists(proxydbpath_data):
-        try:
-            docroot = plist[&quot;DocumentRoot&quot;]
-        except KeyError:
-            &quot;Unable to read DocumentRoot key from plist: %s&quot; % (plistdbpath,)
-            raise
-        proxydbpath_doc = os.path.join(docroot, &quot;principals&quot;, &quot;.db.calendaruserproxy&quot;)
-        if not os.path.exists(proxydbpath_doc):
-            raise(&quot;Unable to find proxy db at '%s' or '%s'&quot; % (proxydbpath_data, proxydbpath_doc))
-        else:
-            proxydbpath = proxydbpath_doc 
-    else:
-        proxydbpath = proxydbpath_data
-    
-    print &quot;&quot;
-    print &quot;Parsed:              %s&quot; % (plistdbpath,)
-    print &quot;Found DS Node:       %s&quot; % (dsnode,)
-    print &quot;Found proxy DB path: %s&quot; % (proxydbpath)
-    print &quot;&quot;
-    return dsnode, proxydbpath
-
-def loadUserRecords(dsnode):
-
-    print &quot;Loading /Users records from OD: %s&quot; % (dsnode,)
-    od = opendirectory.odInit(dsnode)
-    results = opendirectory.listAllRecordsWithAttributes(
-        od,
-        dsattributes.kDSStdRecordTypeUsers,
-        [dsattributes.kDS1AttrGeneratedUID,]
-    )
-
-    result = set()
-    for record in results.itervalues():
-        guid = record.get(dsattributes.kDS1AttrGeneratedUID, None)
-        if guid:
-            result.add(guid)
-
-    print &quot;Found %d /Users records&quot; % (len(result),)
-    print &quot;&quot;
-            
-    return result
-        
-def cleanProxies(existing_guids, proxydbpath, do_changes):
-    
-    # Get proxy entries
-    print &quot;Reading proxy DB: %s&quot; % (proxydbpath,)
-    result = commands.getoutput(&quot;sqlite3 %s \&quot;select * from GROUPS\&quot;&quot; % (proxydbpath,))
-    if not result:
-        print &quot;Proxy DB empty. Nothing to do.&quot;
-        return
-
-    lines = result.split(&quot;\n&quot;)
-    print &quot;Found %d proxy DB records&quot; % (len(lines),)
-
-    proxying = set()
-    proxying_total = 0
-    users = set()
-    users_total = 0
-    for line in lines:
-        proxy, user = line.split(&quot;|&quot;)
-        if proxy.split(&quot;#&quot;)[0] not in existing_guids:
-            proxying.add(proxy)
-            proxying_total += 1
-        if user not in existing_guids:
-            users.add(user)
-            users_total += 1
-
-    if do_changes:
-        print &quot;Changes will be made to the proxy DB&quot;
-    else:
-        print &quot;Changes will not be made to the proxy DB&quot;
-
-    print &quot;&quot;
-    print &quot;Found %d invalid group names (total records %d)&quot; % (len(proxying), proxying_total,)
-    print &quot;============================&quot;
-    for key in sorted(proxying):
-        print key
-        if do_changes:
-            commands.getoutput(&quot;sqlite3 %s \&quot;delete from GROUPS where GROUPNAME = '%s'\&quot;&quot; % (proxydbpath, key,))
-    print &quot;============================&quot;
-    
-    print &quot;&quot;
-    print &quot;Found %d invalid members (total records %d)&quot; % (len(users), users_total,)
-    print &quot;============================&quot;
-    for key in sorted(users):
-        print key
-        if do_changes:
-            commands.getoutput(&quot;sqlite3 %s \&quot;delete from GROUPS where MEMBER = '%s'\&quot;&quot; % (proxydbpath, key,))
-    print &quot;============================&quot;
-    print &quot;&quot;
-    print &quot;Done.&quot;
-
-if __name__ == &quot;__main__&quot;:
-    try:
-        (optargs, args) = getopt.getopt(sys.argv[1:], &quot;cf:h&quot;, [&quot;help&quot;])
-    except getopt.GetoptError, e:
-        usage(e)
-
-    if len(args) != 0:
-        usage(&quot;Wrong number of arguments.&quot;)
-
-    plistdbpath = &quot;/etc/caldavd/caldavd.plist&quot;
-    do_changes = False
-
-    for opt, arg in optargs:
-        if opt in (&quot;-h&quot;, &quot;--help&quot;):
-            usage()
-        elif opt == &quot;-c&quot;:
-            do_changes = True
-        elif opt == &quot;-f&quot;:
-            plistdbpath = arg
-
-    try:
-        print &quot;CalendarServer proxy DB clean-up tool&quot;
-        print &quot;=====================================&quot;
-    
-        if not os.path.exists(plistdbpath):
-            raise ValueError(&quot;caldavd.plist file does not exist: %s&quot; % (plistdbpath,))
-    
-        dsnode, proxydbpath = extractPlistPieces(plistdbpath)
-        guids = loadUserRecords(dsnode)
-        cleanProxies(guids, proxydbpath, do_changes)
-        sys.exit(0)
-    except ValueError, e:
-        print &quot;&quot;
-        print &quot;Failed: %s&quot; % (str(e),)
</del></span></pre>
</div>
</div>

</body>
</html>