2011-06-29 19:43:10 +02:00
|
|
|
/* SOGoSieveManager.m - this file is part of SOGo
|
2010-03-02 20:57:30 +01:00
|
|
|
*
|
2011-06-29 19:43:10 +02:00
|
|
|
* Copyright (C) 2010-2011 Inverse inc.
|
2010-03-02 20:57:30 +01:00
|
|
|
*
|
|
|
|
* Author: Wolfgang Sourdeau <wsourdeau@inverse.ca>
|
2011-06-29 19:43:10 +02:00
|
|
|
* Ludovic Marcotte <lmarcotte@inverse.ca>
|
2010-03-02 20:57:30 +01:00
|
|
|
*
|
|
|
|
* This file 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, or (at your option)
|
|
|
|
* any later version.
|
|
|
|
*
|
|
|
|
* This file 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; see the file COPYING. If not, write to
|
|
|
|
* the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
* Boston, MA 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#import <Foundation/NSArray.h>
|
|
|
|
#import <Foundation/NSDictionary.h>
|
|
|
|
#import <Foundation/NSString.h>
|
2011-06-29 19:43:10 +02:00
|
|
|
#import <Foundation/NSURL.h>
|
2010-03-02 20:57:30 +01:00
|
|
|
#import <Foundation/NSValue.h>
|
|
|
|
|
|
|
|
#import <SOGo/NSArray+Utilities.h>
|
|
|
|
#import <SOGo/NSDictionary+Utilities.h>
|
|
|
|
#import <SOGo/SOGoDomainDefaults.h>
|
|
|
|
#import <SOGo/SOGoUser.h>
|
|
|
|
|
2011-06-29 19:43:10 +02:00
|
|
|
#import <NGStreams/NGInternetSocketAddress.h>
|
|
|
|
#import <NGImap4/NGSieveClient.h>
|
2010-03-02 20:57:30 +01:00
|
|
|
|
2011-06-29 19:43:10 +02:00
|
|
|
#import "../Mailer/SOGoMailAccount.h"
|
|
|
|
|
|
|
|
#import "SOGoSieveManager.h"
|
2010-03-02 20:57:30 +01:00
|
|
|
|
|
|
|
typedef enum {
|
|
|
|
UIxFilterFieldTypeAddress,
|
|
|
|
UIxFilterFieldTypeHeader,
|
|
|
|
UIxFilterFieldTypeSize,
|
|
|
|
} UIxFilterFieldType;
|
|
|
|
|
|
|
|
static NSArray *sieveOperators = nil;
|
|
|
|
static NSArray *sieveSizeOperators = nil;
|
|
|
|
static NSMutableDictionary *fieldTypes = nil;
|
|
|
|
static NSDictionary *sieveFields = nil;
|
|
|
|
static NSDictionary *sieveFlags = nil;
|
|
|
|
static NSDictionary *operatorRequirements = nil;
|
|
|
|
static NSDictionary *methodRequirements = nil;
|
2011-06-29 19:43:10 +02:00
|
|
|
static NSString *sieveScriptName = @"sogo";
|
|
|
|
|
2010-03-02 20:57:30 +01:00
|
|
|
|
|
|
|
@interface NSString (SOGoSieveExtension)
|
|
|
|
|
|
|
|
- (NSString *) asSieveQuotedString;
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation NSString (SOGoSieveExtension)
|
|
|
|
|
|
|
|
- (NSString *) _asSingleLineSieveQuotedString
|
|
|
|
{
|
|
|
|
NSString *escapedString;
|
|
|
|
|
|
|
|
escapedString = [[self stringByReplacingString: @"\\"
|
|
|
|
withString: @"\\\\"]
|
|
|
|
stringByReplacingString: @"\""
|
|
|
|
withString: @"\\\""];
|
|
|
|
|
|
|
|
return [NSString stringWithFormat: @"\"%@\"", escapedString];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString *) _asMultiLineSieveQuotedString
|
|
|
|
{
|
|
|
|
NSArray *lines;
|
|
|
|
NSMutableArray *newLines;
|
|
|
|
NSString *line, *newText;
|
|
|
|
int count, max;
|
|
|
|
|
|
|
|
lines = [self componentsSeparatedByString: @"\n"];
|
|
|
|
max = [lines count];
|
|
|
|
newLines = [NSMutableArray arrayWithCapacity: max];
|
|
|
|
for (count = 0; count < max; count++)
|
|
|
|
{
|
|
|
|
line = [lines objectAtIndex: count];
|
|
|
|
if ([line length] > 0 && [line characterAtIndex: 0] == '.')
|
|
|
|
[newLines addObject: [NSString stringWithFormat: @".%@", line]];
|
|
|
|
else
|
|
|
|
[newLines addObject: line];
|
|
|
|
}
|
|
|
|
|
2011-07-05 18:54:39 +02:00
|
|
|
newText = [NSString stringWithFormat: @"text:\r\n%@\r\n.\r\n",
|
2010-03-02 20:57:30 +01:00
|
|
|
[newLines componentsJoinedByString: @"\n"]];
|
|
|
|
|
|
|
|
return newText;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString *) asSieveQuotedString
|
|
|
|
{
|
|
|
|
NSRange nlRange;
|
|
|
|
|
|
|
|
nlRange = [self rangeOfString: @"\n"];
|
|
|
|
|
|
|
|
return ((nlRange.length > 0)
|
|
|
|
? [self _asMultiLineSieveQuotedString]
|
|
|
|
: [self _asSingleLineSieveQuotedString]);
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
2011-06-29 19:43:10 +02:00
|
|
|
@implementation SOGoSieveManager
|
2010-03-02 20:57:30 +01:00
|
|
|
|
|
|
|
+ (void) initialize
|
|
|
|
{
|
|
|
|
NSArray *fields;
|
|
|
|
|
|
|
|
if (!sieveOperators)
|
|
|
|
{
|
|
|
|
sieveOperators = [NSArray arrayWithObjects: @"is", @"contains",
|
|
|
|
@"matches", @"regex",
|
|
|
|
@"over", @"under", nil];
|
|
|
|
[sieveOperators retain];
|
|
|
|
}
|
|
|
|
if (!sieveSizeOperators)
|
|
|
|
{
|
|
|
|
sieveSizeOperators = [NSArray arrayWithObjects: @"over", @"under", nil];
|
|
|
|
[sieveSizeOperators retain];
|
|
|
|
}
|
|
|
|
if (!fieldTypes)
|
|
|
|
{
|
|
|
|
fieldTypes = [NSMutableDictionary new];
|
|
|
|
fields = [NSArray arrayWithObjects: @"to", @"cc", @"to_or_cc", @"from",
|
|
|
|
nil];
|
|
|
|
[fieldTypes setObject: [NSNumber
|
|
|
|
numberWithInt: UIxFilterFieldTypeAddress]
|
|
|
|
forKeys: fields];
|
|
|
|
fields = [NSArray arrayWithObjects: @"header", @"subject", nil];
|
|
|
|
[fieldTypes setObject: [NSNumber
|
|
|
|
numberWithInt: UIxFilterFieldTypeHeader]
|
|
|
|
forKeys: fields];
|
|
|
|
[fieldTypes setObject: [NSNumber numberWithInt: UIxFilterFieldTypeSize]
|
|
|
|
forKey: @"size"];
|
|
|
|
}
|
|
|
|
if (!sieveFields)
|
|
|
|
{
|
|
|
|
sieveFields
|
|
|
|
= [NSDictionary dictionaryWithObjectsAndKeys:
|
|
|
|
@"\"to\"", @"to",
|
|
|
|
@"\"cc\"", @"cc",
|
|
|
|
@"[\"to\", \"cc\"]", @"to_or_cc",
|
|
|
|
@"\"from\"", @"from",
|
|
|
|
@"\"subject\"", @"subject",
|
|
|
|
nil];
|
|
|
|
[sieveFields retain];
|
|
|
|
}
|
|
|
|
if (!sieveFlags)
|
|
|
|
{
|
|
|
|
sieveFlags
|
|
|
|
= [NSDictionary dictionaryWithObjectsAndKeys:
|
|
|
|
@"\\Answered", @"answered",
|
|
|
|
@"\\Deleted", @"deleted",
|
|
|
|
@"\\Draft", @"draft",
|
|
|
|
@"\\Flagged", @"flagged",
|
|
|
|
@"Junk", @"junk",
|
|
|
|
@"NotJunk", @"not_junk",
|
|
|
|
@"\\Seen", @"seen",
|
|
|
|
@"$Label1", @"label1",
|
|
|
|
@"$Label2", @"label2",
|
|
|
|
@"$Label3", @"label3",
|
|
|
|
@"$Label4", @"label4",
|
|
|
|
@"$Label5", @"label5",
|
|
|
|
nil];
|
|
|
|
[sieveFlags retain];
|
|
|
|
}
|
|
|
|
if (!operatorRequirements)
|
|
|
|
{
|
|
|
|
operatorRequirements
|
|
|
|
= [NSDictionary dictionaryWithObjectsAndKeys:
|
|
|
|
@"regex", @"regex",
|
|
|
|
nil];
|
|
|
|
[operatorRequirements retain];
|
|
|
|
}
|
|
|
|
if (!methodRequirements)
|
|
|
|
{
|
|
|
|
methodRequirements
|
|
|
|
= [NSDictionary dictionaryWithObjectsAndKeys:
|
|
|
|
@"imapflags", @"addflag",
|
|
|
|
@"imapflags", @"removeflag",
|
|
|
|
@"imapflags", @"flag",
|
|
|
|
@"vacation", @"vacation",
|
|
|
|
@"notify", @"notify",
|
|
|
|
@"fileinto", @"fileinto",
|
|
|
|
@"reject", @"reject",
|
|
|
|
@"regex", @"regex",
|
|
|
|
nil];
|
|
|
|
[methodRequirements retain];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-29 19:43:10 +02:00
|
|
|
+ (id) sieveManagerForUser: (SOGoUser *) newUser
|
2010-03-02 20:57:30 +01:00
|
|
|
{
|
2011-06-29 19:43:10 +02:00
|
|
|
SOGoSieveManager *newManager;
|
2010-03-02 20:57:30 +01:00
|
|
|
|
2011-06-29 19:43:10 +02:00
|
|
|
newManager = [[self alloc] initForUser: newUser];
|
|
|
|
[newManager autorelease];
|
2010-03-02 20:57:30 +01:00
|
|
|
|
2011-06-29 19:43:10 +02:00
|
|
|
return newManager;
|
2010-03-02 20:57:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
- (id) init
|
|
|
|
{
|
|
|
|
if ((self = [super init]))
|
|
|
|
{
|
|
|
|
user = nil;
|
|
|
|
requirements = nil;
|
|
|
|
scriptError = nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (id) initForUser: (SOGoUser *) newUser
|
|
|
|
{
|
|
|
|
if ((self = [self init]))
|
|
|
|
{
|
|
|
|
ASSIGN (user, newUser);
|
|
|
|
}
|
|
|
|
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void) dealloc
|
|
|
|
{
|
|
|
|
[user release];
|
|
|
|
[requirements release];
|
|
|
|
[scriptError release];
|
|
|
|
[super dealloc];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL) _saveFilters
|
|
|
|
{
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL) _extractRuleField: (NSString **) field
|
|
|
|
fromRule: (NSDictionary *) rule
|
|
|
|
andType: (UIxFilterFieldType *) type
|
|
|
|
{
|
|
|
|
NSNumber *fieldType;
|
|
|
|
NSString *jsonField, *customHeader;
|
|
|
|
|
|
|
|
jsonField = [rule objectForKey: @"field"];
|
|
|
|
if (jsonField)
|
|
|
|
{
|
|
|
|
fieldType = [fieldTypes objectForKey: jsonField];
|
|
|
|
if (fieldType)
|
|
|
|
{
|
|
|
|
*type = [fieldType intValue];
|
|
|
|
if ([jsonField isEqualToString: @"header"])
|
|
|
|
{
|
|
|
|
customHeader = [rule objectForKey: @"custom_header"];
|
|
|
|
if ([customHeader length])
|
|
|
|
*field = [customHeader asSieveQuotedString];
|
|
|
|
else
|
|
|
|
scriptError = (@"Pseudo-header field 'header' without"
|
|
|
|
@" 'custom_header' parameter.");
|
|
|
|
}
|
|
|
|
else if ([jsonField isEqualToString: @"size"])
|
|
|
|
*field = nil;
|
|
|
|
else
|
|
|
|
*field = [sieveFields objectForKey: jsonField];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
scriptError
|
|
|
|
= [NSString stringWithFormat: @"Rule based on unknown field '%@'",
|
|
|
|
*field];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
scriptError = @"Rule without any specified field.";
|
|
|
|
|
|
|
|
return (scriptError == nil);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL) _extractRuleOperator: (NSString **) operator
|
|
|
|
fromRule: (NSDictionary *) rule
|
|
|
|
isNot: (BOOL *) isNot
|
|
|
|
{
|
|
|
|
NSString *jsonOperator, *baseOperator, *requirement;
|
|
|
|
int baseLength;
|
|
|
|
|
|
|
|
jsonOperator = [rule objectForKey: @"operator"];
|
|
|
|
if (jsonOperator)
|
|
|
|
{
|
|
|
|
*isNot = [jsonOperator hasSuffix: @"_not"];
|
|
|
|
if (*isNot)
|
|
|
|
{
|
|
|
|
baseLength = [jsonOperator length] - 4;
|
|
|
|
baseOperator
|
|
|
|
= [jsonOperator substringWithRange: NSMakeRange (0, baseLength)];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
baseOperator = jsonOperator;
|
|
|
|
|
|
|
|
if ([sieveOperators containsObject: baseOperator])
|
|
|
|
{
|
|
|
|
requirement = [operatorRequirements objectForKey: baseOperator];
|
|
|
|
if (requirement)
|
|
|
|
[requirements addObjectUniquely: requirement];
|
|
|
|
*operator = baseOperator;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
scriptError = [NSString stringWithFormat:
|
|
|
|
@"Rule has unknown operator '%@'",
|
|
|
|
baseOperator];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
scriptError = @"Rule without any specified operator";
|
|
|
|
|
|
|
|
return (scriptError == nil);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL) _validateRuleOperator: (NSString *) operator
|
|
|
|
withFieldType: (UIxFilterFieldType) type
|
|
|
|
{
|
|
|
|
BOOL rc;
|
|
|
|
|
|
|
|
if (type == UIxFilterFieldTypeSize)
|
|
|
|
rc = [sieveSizeOperators containsObject: operator];
|
|
|
|
else
|
|
|
|
rc = (![sieveSizeOperators containsObject: operator]
|
|
|
|
&& [sieveOperators containsObject: operator]);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL) _extractRuleValue: (NSString **) value
|
|
|
|
fromRule: (NSDictionary *) rule
|
|
|
|
withFieldType: (UIxFilterFieldType) type
|
|
|
|
{
|
|
|
|
NSString *extractedValue;
|
|
|
|
|
|
|
|
extractedValue = [rule objectForKey: @"value"];
|
|
|
|
if (extractedValue)
|
|
|
|
{
|
|
|
|
if (type == UIxFilterFieldTypeSize)
|
|
|
|
*value = [NSString stringWithFormat: @"%d",
|
|
|
|
[extractedValue intValue]];
|
|
|
|
else
|
|
|
|
*value = [extractedValue asSieveQuotedString];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
scriptError = @"Rule lacks a 'value' parameter";
|
|
|
|
|
|
|
|
return (scriptError == nil);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString *) _composeSieveRuleOnField: (NSString *) field
|
|
|
|
withType: (UIxFilterFieldType) type
|
|
|
|
operator: (NSString *) operator
|
|
|
|
revert: (BOOL) revert
|
|
|
|
andValue: (NSString *) value
|
|
|
|
{
|
|
|
|
NSMutableString *sieveRule;
|
|
|
|
|
|
|
|
sieveRule = [NSMutableString stringWithCapacity: 100];
|
|
|
|
if (revert)
|
|
|
|
[sieveRule appendString: @"not "];
|
|
|
|
if (type == UIxFilterFieldTypeAddress)
|
|
|
|
[sieveRule appendString: @"address "];
|
|
|
|
else if (type == UIxFilterFieldTypeHeader)
|
|
|
|
[sieveRule appendString: @"header "];
|
|
|
|
else if (type == UIxFilterFieldTypeSize)
|
|
|
|
[sieveRule appendString: @"size "];
|
|
|
|
[sieveRule appendFormat: @":%@ ", operator];
|
|
|
|
if (type == UIxFilterFieldTypeSize)
|
|
|
|
[sieveRule appendFormat: @"%@K", value];
|
|
|
|
else
|
|
|
|
[sieveRule appendFormat: @"%@ %@", field, value];
|
|
|
|
|
|
|
|
return sieveRule;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString *) _extractSieveRule: (NSDictionary *) rule
|
|
|
|
{
|
|
|
|
NSString *field, *operator, *value;
|
|
|
|
UIxFilterFieldType type;
|
|
|
|
BOOL isNot;
|
|
|
|
|
|
|
|
return (([self _extractRuleField: &field fromRule: rule andType: &type]
|
|
|
|
&& [self _extractRuleOperator: &operator fromRule: rule
|
|
|
|
isNot: &isNot]
|
|
|
|
&& [self _validateRuleOperator: operator
|
|
|
|
withFieldType: type]
|
|
|
|
&& [self _extractRuleValue: &value fromRule: rule
|
|
|
|
withFieldType: type])
|
|
|
|
? [self _composeSieveRuleOnField: field
|
|
|
|
withType: type
|
|
|
|
operator: operator
|
|
|
|
revert: isNot
|
|
|
|
andValue: value]
|
|
|
|
: nil);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSArray *) _extractSieveRules: (NSArray *) rules
|
|
|
|
{
|
|
|
|
NSMutableArray *sieveRules;
|
|
|
|
NSString *sieveRule;
|
|
|
|
int count, max;
|
|
|
|
|
|
|
|
max = [rules count];
|
|
|
|
if (max)
|
|
|
|
{
|
|
|
|
sieveRules = [NSMutableArray arrayWithCapacity: max];
|
|
|
|
for (count = 0; !scriptError && count < max; count++)
|
|
|
|
{
|
|
|
|
sieveRule = [self _extractSieveRule: [rules objectAtIndex: count]];
|
|
|
|
if (sieveRule)
|
|
|
|
[sieveRules addObject: sieveRule];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
sieveRules = nil;
|
|
|
|
|
|
|
|
return sieveRules;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString *) _extractSieveAction: (NSDictionary *) action
|
|
|
|
{
|
|
|
|
NSString *sieveAction, *method, *requirement, *argument,
|
|
|
|
*flag, *mailbox;
|
|
|
|
SOGoDomainDefaults *dd;
|
|
|
|
|
2010-04-21 16:35:58 +02:00
|
|
|
sieveAction = nil;
|
|
|
|
|
2010-03-02 20:57:30 +01:00
|
|
|
method = [action objectForKey: @"method"];
|
|
|
|
if (method)
|
|
|
|
{
|
|
|
|
argument = [action objectForKey: @"argument"];
|
|
|
|
if ([method isEqualToString: @"discard"]
|
|
|
|
|| [method isEqualToString: @"keep"]
|
|
|
|
|| [method isEqualToString: @"stop"])
|
|
|
|
sieveAction = method;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (argument)
|
|
|
|
{
|
|
|
|
if ([method isEqualToString: @"addflag"])
|
|
|
|
{
|
|
|
|
flag = [sieveFlags objectForKey: argument];
|
|
|
|
if (flag)
|
|
|
|
sieveAction = [NSString stringWithFormat: @"%@ %@",
|
|
|
|
method, [flag asSieveQuotedString]];
|
|
|
|
else
|
|
|
|
scriptError
|
|
|
|
= [NSString stringWithFormat:
|
|
|
|
@"Action with invalid flag argument '%@'",
|
|
|
|
argument];
|
|
|
|
}
|
|
|
|
else if ([method isEqualToString: @"fileinto"])
|
|
|
|
{
|
|
|
|
dd = [user domainDefaults];
|
|
|
|
mailbox
|
2011-02-03 19:49:48 +01:00
|
|
|
= [[argument componentsSeparatedByString: @"/"]
|
2010-03-02 20:57:30 +01:00
|
|
|
componentsJoinedByString: [dd imapFolderSeparator]];
|
|
|
|
sieveAction = [NSString stringWithFormat: @"%@ %@",
|
|
|
|
method, [mailbox asSieveQuotedString]];
|
|
|
|
}
|
|
|
|
else if ([method isEqualToString: @"redirect"])
|
|
|
|
sieveAction = [NSString stringWithFormat: @"%@ %@",
|
|
|
|
method, [argument asSieveQuotedString]];
|
|
|
|
else if ([method isEqualToString: @"reject"])
|
2011-07-05 18:54:39 +02:00
|
|
|
sieveAction = [NSString stringWithFormat: @"%@ %@",
|
|
|
|
method, [argument asSieveQuotedString]];
|
2010-03-02 20:57:30 +01:00
|
|
|
else
|
|
|
|
scriptError
|
|
|
|
= [NSString stringWithFormat: @"Action has unknown method '%@'",
|
|
|
|
method];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
scriptError = @"Action missing 'argument' parameter";
|
|
|
|
}
|
|
|
|
if (method)
|
|
|
|
{
|
|
|
|
requirement = [methodRequirements objectForKey: method];
|
|
|
|
if (requirement)
|
|
|
|
[requirements addObjectUniquely: requirement];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2010-04-21 16:35:58 +02:00
|
|
|
scriptError = @"Action missing 'method' parameter";
|
2010-03-02 20:57:30 +01:00
|
|
|
|
|
|
|
return sieveAction;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSArray *) _extractSieveActions: (NSArray *) actions
|
|
|
|
{
|
|
|
|
NSMutableArray *sieveActions;
|
|
|
|
NSString *sieveAction;
|
|
|
|
int count, max;
|
|
|
|
|
|
|
|
max = [actions count];
|
|
|
|
sieveActions = [NSMutableArray arrayWithCapacity: max];
|
|
|
|
for (count = 0; !scriptError && count < max; count++)
|
|
|
|
{
|
|
|
|
sieveAction = [self _extractSieveAction: [actions objectAtIndex: count]];
|
|
|
|
if (!scriptError)
|
|
|
|
[sieveActions addObject: sieveAction];
|
|
|
|
}
|
|
|
|
|
|
|
|
return sieveActions;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString *) _convertScriptToSieve: (NSDictionary *) newScript
|
|
|
|
{
|
|
|
|
NSMutableString *sieveText;
|
|
|
|
NSString *match;
|
|
|
|
NSArray *sieveRules, *sieveActions;
|
|
|
|
|
|
|
|
sieveText = [NSMutableString stringWithCapacity: 1024];
|
|
|
|
match = [newScript objectForKey: @"match"];
|
|
|
|
if ([match isEqualToString: @"allmessages"])
|
|
|
|
match = nil;
|
|
|
|
if (match)
|
|
|
|
{
|
|
|
|
if ([match isEqualToString: @"all"] || [match isEqualToString: @"any"])
|
|
|
|
{
|
|
|
|
sieveRules
|
|
|
|
= [self _extractSieveRules: [newScript objectForKey: @"rules"]];
|
|
|
|
if (sieveRules)
|
|
|
|
[sieveText appendFormat: @"if %@of (%@) {\r\n",
|
|
|
|
match,
|
|
|
|
[sieveRules componentsJoinedByString: @", "]];
|
|
|
|
else
|
|
|
|
scriptError = [NSString stringWithFormat:
|
|
|
|
@"Test '%@' used without any"
|
|
|
|
@" specified rule",
|
|
|
|
match];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
scriptError = [NSString stringWithFormat: @"Bad test: %@",
|
|
|
|
match];
|
|
|
|
}
|
|
|
|
sieveActions = [self _extractSieveActions:
|
|
|
|
[newScript objectForKey: @"actions"]];
|
|
|
|
if ([sieveActions count])
|
|
|
|
[sieveText appendFormat: @" %@;\r\n",
|
|
|
|
[sieveActions componentsJoinedByString: @";\r\n "]];
|
|
|
|
|
|
|
|
if (match)
|
|
|
|
[sieveText appendFormat: @"}\r\n"];
|
|
|
|
|
|
|
|
return sieveText;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString *) sieveScriptWithRequirements: (NSMutableArray *) newRequirements
|
|
|
|
{
|
|
|
|
NSMutableString *sieveScript;
|
|
|
|
NSString *sieveText;
|
|
|
|
NSArray *scripts;
|
|
|
|
int count, max;
|
|
|
|
BOOL previousWasConditional;
|
|
|
|
NSDictionary *currentScript;
|
|
|
|
|
|
|
|
sieveScript = [NSMutableString stringWithCapacity: 8192];
|
|
|
|
|
|
|
|
ASSIGN (requirements, newRequirements);
|
|
|
|
[scriptError release];
|
|
|
|
scriptError = nil;
|
|
|
|
|
|
|
|
scripts = [[user userDefaults] sieveFilters];
|
|
|
|
max = [scripts count];
|
|
|
|
if (max)
|
|
|
|
{
|
|
|
|
previousWasConditional = NO;
|
|
|
|
for (count = 0; !scriptError && count < max; count++)
|
|
|
|
{
|
|
|
|
currentScript = [scripts objectAtIndex: count];
|
|
|
|
if ([[currentScript objectForKey: @"active"] boolValue])
|
|
|
|
{
|
|
|
|
sieveText = [self _convertScriptToSieve: currentScript];
|
|
|
|
if ([sieveText hasPrefix: @"if"])
|
|
|
|
{
|
|
|
|
if (previousWasConditional)
|
|
|
|
[sieveScript appendFormat: @"els"];
|
|
|
|
else
|
|
|
|
previousWasConditional = YES;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
previousWasConditional = NO;
|
|
|
|
[sieveScript appendString: sieveText];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
[scriptError retain];
|
|
|
|
[requirements release];
|
|
|
|
requirements = nil;
|
|
|
|
|
|
|
|
if (scriptError)
|
|
|
|
sieveScript = nil;
|
|
|
|
|
|
|
|
return sieveScript;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString *) lastScriptError
|
|
|
|
{
|
|
|
|
return scriptError;
|
|
|
|
}
|
|
|
|
|
2011-06-29 19:43:10 +02:00
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
- (BOOL) updateFiltersForLogin: (NSString *) theLogin
|
|
|
|
authname: (NSString *) theAuthName
|
|
|
|
password: (NSString *) thePassword
|
|
|
|
account: (SOGoMailAccount *) theAccount
|
|
|
|
{
|
|
|
|
NSMutableArray *req;
|
|
|
|
NSMutableString *script, *header;
|
|
|
|
NSDictionary *result, *values;
|
|
|
|
SOGoUserDefaults *ud;
|
|
|
|
SOGoDomainDefaults *dd;
|
|
|
|
NGSieveClient *client;
|
|
|
|
NSString *filterScript, *v, *sieveServer;
|
2011-12-15 20:37:02 +01:00
|
|
|
NSURL *url;
|
|
|
|
|
2011-06-29 19:43:10 +02:00
|
|
|
int sievePort;
|
2011-11-17 20:16:46 +01:00
|
|
|
BOOL b, connected;
|
2011-06-29 19:43:10 +02:00
|
|
|
|
|
|
|
dd = [user domainDefaults];
|
|
|
|
if (!([dd sieveScriptsEnabled] || [dd vacationEnabled] || [dd forwardEnabled]))
|
|
|
|
return YES;
|
|
|
|
|
|
|
|
req = [NSMutableArray arrayWithCapacity: 15];
|
|
|
|
ud = [user userDefaults];
|
2011-11-17 20:16:46 +01:00
|
|
|
|
|
|
|
connected = YES;
|
2011-06-29 19:43:10 +02:00
|
|
|
b = NO;
|
|
|
|
|
|
|
|
script = [NSMutableString string];
|
|
|
|
|
|
|
|
// Right now, we handle Sieve filters here and only for vacation
|
|
|
|
// and forwards. Traditional filters support (for fileinto, for
|
|
|
|
// example) will be added later.
|
|
|
|
values = [ud vacationOptions];
|
|
|
|
|
|
|
|
// We handle vacation messages.
|
|
|
|
// See http://ietfreport.isoc.org/idref/draft-ietf-sieve-vacation/
|
|
|
|
if (values && [[values objectForKey: @"enabled"] boolValue])
|
|
|
|
{
|
|
|
|
NSArray *addresses;
|
|
|
|
NSString *text;
|
|
|
|
BOOL ignore;
|
|
|
|
int days, i;
|
|
|
|
|
|
|
|
days = [[values objectForKey: @"daysBetweenResponse"] intValue];
|
|
|
|
addresses = [values objectForKey: @"autoReplyEmailAddresses"];
|
|
|
|
ignore = [[values objectForKey: @"ignoreLists"] boolValue];
|
|
|
|
text = [values objectForKey: @"autoReplyText"];
|
|
|
|
b = YES;
|
|
|
|
|
|
|
|
if (days == 0)
|
|
|
|
days = 7;
|
|
|
|
|
|
|
|
[req addObjectUniquely: @"vacation"];
|
|
|
|
|
|
|
|
// Skip mailing lists
|
|
|
|
if (ignore)
|
|
|
|
[script appendString: @"if allof ( not exists [\"list-help\", \"list-unsubscribe\", \"list-subscribe\", \"list-owner\", \"list-post\", \"list-archive\", \"list-id\", \"Mailing-List\"], not header :comparator \"i;ascii-casemap\" :is \"Precedence\" [\"list\", \"bulk\", \"junk\"], not header :comparator \"i;ascii-casemap\" :matches \"To\" \"Multiple recipients of*\" ) {"];
|
|
|
|
|
|
|
|
[script appendFormat: @"vacation :days %d :addresses [", days];
|
|
|
|
|
|
|
|
for (i = 0; i < [addresses count]; i++)
|
|
|
|
{
|
|
|
|
[script appendFormat: @"\"%@\"", [addresses objectAtIndex: i]];
|
|
|
|
|
|
|
|
if (i == [addresses count]-1)
|
|
|
|
[script appendString: @"] "];
|
|
|
|
else
|
|
|
|
[script appendString: @", "];
|
|
|
|
}
|
|
|
|
|
|
|
|
[script appendFormat: @"text:\r\n%@\r\n.\r\n;\r\n", text];
|
|
|
|
|
|
|
|
if (ignore)
|
|
|
|
[script appendString: @"}\r\n"];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// We handle mail forward
|
|
|
|
values = [ud forwardOptions];
|
|
|
|
|
|
|
|
if (values && [[values objectForKey: @"enabled"] boolValue])
|
|
|
|
{
|
|
|
|
id addresses;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
b = YES;
|
|
|
|
|
|
|
|
addresses = [values objectForKey: @"forwardAddress"];
|
|
|
|
if ([addresses isKindOfClass: [NSString class]])
|
|
|
|
addresses = [NSArray arrayWithObject: addresses];
|
|
|
|
|
|
|
|
for (i = 0; i < [addresses count]; i++)
|
|
|
|
{
|
|
|
|
v = [addresses objectAtIndex: i];
|
|
|
|
if (v && [v length] > 0)
|
|
|
|
[script appendFormat: @"redirect \"%@\";\r\n", v];
|
|
|
|
}
|
|
|
|
|
|
|
|
if ([[values objectForKey: @"keepCopy"] boolValue])
|
|
|
|
[script appendString: @"keep;\r\n"];
|
|
|
|
}
|
|
|
|
|
|
|
|
filterScript = [self sieveScriptWithRequirements: req];
|
|
|
|
if (filterScript)
|
|
|
|
{
|
|
|
|
if ([filterScript length])
|
|
|
|
{
|
|
|
|
b = YES;
|
|
|
|
[script appendString: filterScript];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
NSLog(@"Sieve generation failure: %@", [self lastScriptError]);
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ([req count])
|
|
|
|
{
|
|
|
|
header = [NSString stringWithFormat: @"require [\"%@\"];\r\n",
|
|
|
|
[req componentsJoinedByString: @"\",\""]];
|
|
|
|
[script insertString: header atIndex: 0];
|
|
|
|
}
|
|
|
|
|
|
|
|
// We connect to our Sieve server and upload the script.
|
|
|
|
//
|
|
|
|
// sieveServer might have the following format:
|
|
|
|
//
|
|
|
|
// sieve://localhost
|
|
|
|
// sieve://localhost:2000
|
2011-12-15 20:37:02 +01:00
|
|
|
// sieve://localhost:2000/?tls=YES
|
2011-06-29 19:43:10 +02:00
|
|
|
//
|
|
|
|
// Values such as "localhost" or "localhost:2000" are NOT supported.
|
|
|
|
//
|
|
|
|
sieveServer = [dd sieveServer];
|
|
|
|
sievePort = 2000;
|
2011-12-15 20:37:02 +01:00
|
|
|
url = nil;
|
2011-06-29 19:43:10 +02:00
|
|
|
|
|
|
|
if (!sieveServer)
|
|
|
|
{
|
|
|
|
NSString *s;
|
2011-12-15 20:37:02 +01:00
|
|
|
|
2011-06-29 19:43:10 +02:00
|
|
|
s = [dd imapServer];
|
2011-12-15 20:37:02 +01:00
|
|
|
|
2011-06-29 19:43:10 +02:00
|
|
|
if (s)
|
|
|
|
{
|
|
|
|
NSURL *url;
|
|
|
|
|
|
|
|
url = [NSURL URLWithString: s];
|
|
|
|
|
|
|
|
if ([url host])
|
|
|
|
sieveServer = [url host];
|
|
|
|
else
|
|
|
|
sieveServer = s;
|
|
|
|
}
|
2011-12-15 20:37:02 +01:00
|
|
|
else
|
|
|
|
sieveServer = @"localhost";
|
|
|
|
|
|
|
|
url = [NSURL URLWithString: [NSString stringWithFormat: @"%@:%d", sieveServer, sievePort]];
|
2011-06-29 19:43:10 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
url = [NSURL URLWithString: sieveServer];
|
|
|
|
}
|
|
|
|
|
2011-12-15 20:37:02 +01:00
|
|
|
client = [[NGSieveClient alloc] initWithURL: url];
|
2011-06-29 19:43:10 +02:00
|
|
|
|
|
|
|
if (!client) {
|
2011-12-15 20:37:02 +01:00
|
|
|
NSLog(@"Sieve connection failed on %@", [url description]);
|
2011-06-29 19:43:10 +02:00
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!thePassword) {
|
|
|
|
[client closeConnection];
|
|
|
|
return NO;
|
|
|
|
}
|
2011-11-17 20:16:46 +01:00
|
|
|
|
|
|
|
NS_DURING
|
|
|
|
{
|
|
|
|
result = [client login: theLogin authname: theAuthName password: thePassword];
|
|
|
|
}
|
|
|
|
NS_HANDLER
|
|
|
|
{
|
|
|
|
connected = NO;
|
|
|
|
}
|
|
|
|
NS_ENDHANDLER
|
|
|
|
|
|
|
|
if (!connected)
|
|
|
|
{
|
2011-12-15 20:37:02 +01:00
|
|
|
NSLog(@"Sieve connection failed on %@", [url description]);
|
2011-11-17 20:16:46 +01:00
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
2011-06-29 19:43:10 +02:00
|
|
|
if (![[result valueForKey:@"result"] boolValue]) {
|
|
|
|
NSLog(@"failure. Attempting with a renewed password (no authname supported)");
|
|
|
|
thePassword = [theAccount imap4PasswordRenewed: YES];
|
|
|
|
result = [client login: theLogin password: thePassword];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (![[result valueForKey:@"result"] boolValue]) {
|
|
|
|
NSLog(@"Could not login '%@' on Sieve server: %@: %@",
|
|
|
|
theLogin, client, result);
|
|
|
|
[client closeConnection];
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We ensure to deactive the current active script since it could prevent
|
|
|
|
its deletion from the server. */
|
|
|
|
result = [client setActiveScript: @""];
|
|
|
|
// We delete the existing Sieve script
|
|
|
|
result = [client deleteScript: sieveScriptName];
|
|
|
|
|
|
|
|
if (![[result valueForKey:@"result"] boolValue]) {
|
|
|
|
NSLog(@"WARNING: Could not delete Sieve script - continuing...: %@", result);
|
|
|
|
}
|
|
|
|
|
|
|
|
// We put and activate the script only if we actually have a script
|
|
|
|
// that does something...
|
|
|
|
if (b)
|
|
|
|
{
|
|
|
|
result = [client putScript: sieveScriptName script: script];
|
|
|
|
|
|
|
|
if (![[result valueForKey:@"result"] boolValue]) {
|
|
|
|
NSLog(@"Could not upload Sieve script: %@", result);
|
|
|
|
[client closeConnection];
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = [client setActiveScript: sieveScriptName];
|
|
|
|
if (![[result valueForKey:@"result"] boolValue]) {
|
|
|
|
NSLog(@"Could not enable Sieve script: %@", result);
|
|
|
|
[client closeConnection];
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
|
2010-03-02 20:57:30 +01:00
|
|
|
@end
|