Skip to content

Commit

Permalink
fix(i18n): new localized strings for SQL-based password policies
Browse files Browse the repository at this point in the history
  • Loading branch information
WoodySlum authored Jul 26, 2022
1 parent a667c69 commit 5e6ad77
Show file tree
Hide file tree
Showing 9 changed files with 196 additions and 3 deletions.
3 changes: 3 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -28,3 +28,6 @@ UI/WebServerResources/scss/.sass-cache/
config.make
doc
tags
.DS_Store
Tests/package-lock.json
.vscode
27 changes: 27 additions & 0 deletions Documentation/SOGoInstallationGuide.asciidoc
Original file line number Diff line number Diff line change
Expand Up @@ -1694,6 +1694,33 @@ userPasswordPolicy = (
);
----
Pre-defined constants can also be used :
----
userPasswordPolicy = (
{
label = "POLICY_MIN_LOWERCASE_LETTER";
value = 1;
},
{
label = "POLICY_MIN_UPPERCASE_LETTER";
value = 1;
},
{
label = "POLICY_MIN_DIGIT";
value = 2;
},
{
label = "POLICY_MIN_SPECIAL_SYMBOLS";
value = 1;
},
{
label = "POLICY_MIN_LENGTH";
value = 8;
}
);
----
|userPasswordAlgorithm
|The default algorithm used for password encryption when changing
passwords. Possible values are: `none`, `plain`, `crypt`, `md5`,
Expand Down
1 change: 1 addition & 0 deletions SoObjects/SOGo/GNUmakefile
Original file line number Diff line number Diff line change
Expand Up @@ -129,6 +129,7 @@ SOGo_OBJC_FILES = \
SOGoUserManager.m \
LDAPSource.m \
LDAPSourceSchema.m \
SOGoPasswordPolicy.m \
SQLSource.m \
SOGoUserProfile.m \
SOGoSQLUserProfile.m \
Expand Down
19 changes: 19 additions & 0 deletions SoObjects/SOGo/SOGoPasswordPolicy.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
#ifndef SOGO_PASSWORD_POLICY_H
#define SOGO_PASSWORD_POLICY_H

#import <Foundation/NSObject.h>

@interface SOGoPasswordPolicy : NSObject
{

}

+ (NSArray *) policies;
+ (NSArray *) regexPoliciesWithCount:(NSNumber *) count;
+ (NSArray *) createPasswordPolicyRegex: (NSArray *) userPasswordPolicy;
+ (NSArray *) createPasswordPolicyLabels: (NSArray *) userPasswordPolicy
withTranslations: (NSDictionary *) translations;

@end

#endif /* SOGO_PASSWORD_POLICY_H */
118 changes: 118 additions & 0 deletions SoObjects/SOGo/SOGoPasswordPolicy.m
Original file line number Diff line number Diff line change
@@ -0,0 +1,118 @@
/* SOGoPasswordPolicy.h - this file is part of SOGo
*
* Copyright (C) 2009-2022 Alinto
*
* This file is part of SOGo.
*
* 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/NSDictionary.h>
#import <Foundation/NSArray.h>
#import <Foundation/NSValue.h>
#import <Foundation/NSException.h>

#import "SOGoPasswordPolicy.h"

static const NSString *POLICY_MIN_LOWERCASE_LETTER = @"POLICY_MIN_LOWERCASE_LETTER";
static const NSString *POLICY_MIN_UPPERCASE_LETTER = @"POLICY_MIN_UPPERCASE_LETTER";
static const NSString *POLICY_MIN_DIGIT = @"POLICY_MIN_DIGIT";
static const NSString *POLICY_MIN_SPECIAL_SYMBOLS = @"POLICY_MIN_SPECIAL_SYMBOLS";
static const NSString *POLICY_MIN_LENGTH = @"POLICY_MIN_LENGTH";

@implementation SOGoPasswordPolicy

- (id) init
{
return [super init];
}

- (void) dealloc
{
[super dealloc];
}

+ (NSArray *) policies {
return [NSArray arrayWithObjects: POLICY_MIN_LOWERCASE_LETTER,
POLICY_MIN_UPPERCASE_LETTER,
POLICY_MIN_DIGIT,
POLICY_MIN_SPECIAL_SYMBOLS,
POLICY_MIN_LENGTH,
nil];
}

+ (NSArray *) regexPoliciesWithCount:(NSNumber *) count {
return [NSArray arrayWithObjects: [NSString stringWithFormat:@"(.*[a-z].*){%i}", [count intValue]],
[NSString stringWithFormat:@"(.*[A-Z].*){%i}", [count intValue]],
[NSString stringWithFormat:@"(.*[0-9].*){%i}", [count intValue]],
[NSString stringWithFormat:@"([%$&*(){}!?\\@#].*){%i,}", [count intValue]],
[NSString stringWithFormat:@".{%i,}", [count intValue]],
nil];
}

+ (NSArray *) createPasswordPolicyRegex: (NSArray *) userPasswordPolicy
{
NSMutableArray *passwordPolicy = [[NSMutableArray alloc] init];
[passwordPolicy autorelease];
for (NSDictionary *policy in userPasswordPolicy) {
NSString *label = [policy objectForKey:@"label"];
if ([[self policies] containsObject: label]) {
NSNumber *value = [policy objectForKey:@"value"];
NSInteger index = [[self policies] indexOfObject: label];

if (0 < value) {
NSMutableDictionary *newPolicy = [NSMutableDictionary dictionaryWithDictionary: policy];
[newPolicy setObject:[[self regexPoliciesWithCount: value] objectAtIndex: index] forKey:@"regex"];
[passwordPolicy addObject: newPolicy];
} else {
// Do nothing
}
} else {
[passwordPolicy addObject: policy];
}
}
return passwordPolicy;
}

+ (NSArray *) createPasswordPolicyLabels: (NSArray *) userPasswordPolicy
withTranslations: (NSDictionary *) translations
{
NSMutableArray *userTranslatedPasswordPolicy = [[NSMutableArray alloc] init];
[userTranslatedPasswordPolicy autorelease];
for (NSDictionary *policy in userPasswordPolicy) {
NSString *label = [policy objectForKey:@"label"];
if ([[self policies] containsObject: label]) {
NSNumber *value = [policy objectForKey:@"value"];
if (0 < value) {
NSString *newLabel = [[translations objectForKey: label]
stringByReplacingOccurrencesOfString: @"%{0}"
withString: [value stringValue]];
[userTranslatedPasswordPolicy addObject:[NSDictionary dictionaryWithObjectsAndKeys:
newLabel, @"label",
[policy objectForKey:@"regex"], @"regex",
nil]];
} else {
// Do nothing
}
} else {
[userTranslatedPasswordPolicy addObject: policy];
}
}

return userTranslatedPasswordPolicy;
}

@end
3 changes: 2 additions & 1 deletion SoObjects/SOGo/SQLSource.m
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@
#import "NSString+Crypto.h"

#import "SQLSource.h"
#import "SOGoPasswordPolicy.h"

/**
* The view MUST contain the following columns:
Expand Down Expand Up @@ -145,7 +146,7 @@ - (id) initFromUDSource: (NSDictionary *) udSource
ASSIGN(_authenticationFilter, [udSource objectForKey: @"authenticationFilter"]);
ASSIGN(_loginFields, [udSource objectForKey: @"LoginFieldNames"]);
ASSIGN(_mailFields, [udSource objectForKey: @"MailFieldNames"]);
ASSIGN(_userPasswordPolicy, [udSource objectForKey: @"userPasswordPolicy"]);
ASSIGN(_userPasswordPolicy, [SOGoPasswordPolicy createPasswordPolicyRegex: [udSource objectForKey: @"userPasswordPolicy"]]);
ASSIGN(_userPasswordAlgorithm, [udSource objectForKey: @"userPasswordAlgorithm"]);
ASSIGN(_keyPath, [udSource objectForKey: @"keyPath"]);
ASSIGN(_imapLoginField, [udSource objectForKey: @"IMAPLoginFieldName"]);
Expand Down
6 changes: 6 additions & 0 deletions UI/PreferencesUI/English.lproj/Localizable.strings
Original file line number Diff line number Diff line change
Expand Up @@ -257,6 +257,12 @@
"Confirmation" = "Confirmation";
"Change" = "Change";
"Passwords don't match" = "Passwords don't match";
"POLICY_MIN_LOWERCASE_LETTER" = "Minimum of %{0} lowercase letter";
"POLICY_MIN_UPPERCASE_LETTER" = "Minimum of %{0} uppercase letter";
"POLICY_MIN_DIGIT" = "Minimum of %{0} digit";
"POLICY_MIN_SPECIAL_SYMBOLS" = "Minimum of %{0}special symbols";
"POLICY_MIN_LENGTH" = "Minimum length of %{0} characters";


/* Event+task classifications */
"Default events classification" = "Default events classification";
Expand Down
5 changes: 5 additions & 0 deletions UI/PreferencesUI/French.lproj/Localizable.strings
Original file line number Diff line number Diff line change
Expand Up @@ -257,6 +257,11 @@
"Confirmation" = "Confirmation";
"Change" = "Changer";
"Passwords don't match" = "Les mots de passe ne correspondent pas";
"POLICY_MIN_LOWERCASE_LETTER" = "Au moins %{0} lettre(s) minuscule(s)";
"POLICY_MIN_UPPERCASE_LETTER" = "Au moins %{0} lettre(s) majuscule(s)";
"POLICY_MIN_DIGIT" = "Au moins %{0} chiffre(s)";
"POLICY_MIN_SPECIAL_SYMBOLS" = "Au moins %{0} caractère(s) special(aux)";
"POLICY_MIN_LENGTH" = "Longueur d'au moins %{0} caractère(s)";

/* Event+task classifications */
"Default events classification" = "Classification par défaut des événements";
Expand Down
17 changes: 15 additions & 2 deletions UI/PreferencesUI/UIxPreferences.m
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@
#import <SOGo/SOGoTextTemplateFile.h>
#import <SOGo/WOResourceManager+SOGo.h>
#import <SOGo/SOGoBuild.h>
#import <SOGo/SOGoPasswordPolicy.h>
#import <Mailer/SOGoMailAccount.h>
#import <Mailer/SOGoMailAccounts.h>

Expand Down Expand Up @@ -257,9 +258,21 @@ - (NSString *) itemShortDateFormatText
- (NSArray *) passwordPolicy
{
NSObject <SOGoSource> *userSource;

NSMutableDictionary *translations = [[NSMutableDictionary alloc] init];
NSDictionary *policy;
NSDictionary *translatedUserPolicy;

userSource = [user authenticationSource];
return [userSource userPasswordPolicy];

for(policy in [userSource userPasswordPolicy]) {
[translations setObject:[self labelForKey:[policy objectForKey:@"label"]]
forKey: [policy objectForKey:@"label"]];
}
translatedUserPolicy = [SOGoPasswordPolicy createPasswordPolicyLabels: [userSource userPasswordPolicy]
withTranslations: translations];
[translations release];

return translatedUserPolicy;
}

//
Expand Down

0 comments on commit 5e6ad77

Please sign in to comment.