mirror of
				https://github.com/steveiliop56/tinyauth.git
				synced 2025-11-04 08:05:42 +00:00 
			
		
		
		
	Compare commits
	
		
			6 Commits
		
	
	
		
			v3.5.0
			...
			9ed254cbe3
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					9ed254cbe3 | ||
| 
						 | 
					fe440a6f2e | ||
| 
						 | 
					0ace88a877 | ||
| 
						 | 
					476ed6964d | ||
| 
						 | 
					b3dca0429f | ||
| 
						 | 
					9e4b68112c | 
@@ -53,7 +53,7 @@ Tinyauth is licensed under the GNU General Public License v3.0. TL;DR — You ma
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
A big thank you to the following people for providing me with more coffee:
 | 
					A big thank you to the following people for providing me with more coffee:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
<!-- sponsors --><a href="https://github.com/erwinkramer"><img src="https://github.com/erwinkramer.png" width="64px" alt="User avatar: erwinkramer" /></a>  <a href="https://github.com/nicotsx"><img src="https://github.com/nicotsx.png" width="64px" alt="User avatar: nicotsx" /></a>  <a href="https://github.com/SimpleHomelab"><img src="https://github.com/SimpleHomelab.png" width="64px" alt="User avatar: SimpleHomelab" /></a>  <a href="https://github.com/jmadden91"><img src="https://github.com/jmadden91.png" width="64px" alt="User avatar: jmadden91" /></a>  <a href="https://github.com/tribor"><img src="https://github.com/tribor.png" width="64px" alt="User avatar: tribor" /></a>  <a href="https://github.com/eliasbenb"><img src="https://github.com/eliasbenb.png" width="64px" alt="User avatar: eliasbenb" /></a>  <!-- sponsors -->
 | 
					<!-- sponsors --><a href="https://github.com/erwinkramer"><img src="https://github.com/erwinkramer.png" width="64px" alt="User avatar: erwinkramer" /></a>  <a href="https://github.com/nicotsx"><img src="https://github.com/nicotsx.png" width="64px" alt="User avatar: nicotsx" /></a>  <a href="https://github.com/SimpleHomelab"><img src="https://github.com/SimpleHomelab.png" width="64px" alt="User avatar: SimpleHomelab" /></a>  <a href="https://github.com/jmadden91"><img src="https://github.com/jmadden91.png" width="64px" alt="User avatar: jmadden91" /></a>  <a href="https://github.com/tribor"><img src="https://github.com/tribor.png" width="64px" alt="User avatar: tribor" /></a>  <a href="https://github.com/eliasbenb"><img src="https://github.com/eliasbenb.png" width="64px" alt="User avatar: eliasbenb" /></a>  <a href="https://github.com/afunworm"><img src="https://github.com/afunworm.png" width="64px" alt="User avatar: afunworm" /></a>  <!-- sponsors -->
 | 
				
			||||||
 | 
					
 | 
				
			||||||
## Acknowledgements
 | 
					## Acknowledgements
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -27,8 +27,8 @@ export const languages = {
 | 
				
			|||||||
  "tr-TR": "Türkçe",
 | 
					  "tr-TR": "Türkçe",
 | 
				
			||||||
  "uk-UA": "Українська",
 | 
					  "uk-UA": "Українська",
 | 
				
			||||||
  "vi-VN": "Tiếng Việt",
 | 
					  "vi-VN": "Tiếng Việt",
 | 
				
			||||||
  "zh-CN": "中文",
 | 
					  "zh-CN": "简体中文",
 | 
				
			||||||
  "zh-TW": "中文",
 | 
					  "zh-TW": "繁體中文(台灣)",
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export type SupportedLanguage = keyof typeof languages;
 | 
					export type SupportedLanguage = keyof typeof languages;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -19,7 +19,7 @@
 | 
				
			|||||||
    "continueInvalidRedirectTitle": "إعادة توجيه غير صالحة",
 | 
					    "continueInvalidRedirectTitle": "إعادة توجيه غير صالحة",
 | 
				
			||||||
    "continueInvalidRedirectSubtitle": "رابط إعادة التوجيه غير صالح",
 | 
					    "continueInvalidRedirectSubtitle": "رابط إعادة التوجيه غير صالح",
 | 
				
			||||||
    "continueInsecureRedirectTitle": "إعادة توجيه غير آمنة",
 | 
					    "continueInsecureRedirectTitle": "إعادة توجيه غير آمنة",
 | 
				
			||||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
					    "continueInsecureRedirectSubtitle": "أنت تحاول إعادة التوجيه من <code>https</code> إلى <code>http</code>، هل أنت متأكد أنك تريد المتابعة؟",
 | 
				
			||||||
    "continueTitle": "متابعة",
 | 
					    "continueTitle": "متابعة",
 | 
				
			||||||
    "continueSubtitle": "انقر الزر للمتابعة إلى التطبيق الخاص بك.",
 | 
					    "continueSubtitle": "انقر الزر للمتابعة إلى التطبيق الخاص بك.",
 | 
				
			||||||
    "logoutFailTitle": "فشل تسجيل الخروج",
 | 
					    "logoutFailTitle": "فشل تسجيل الخروج",
 | 
				
			||||||
@@ -45,7 +45,7 @@
 | 
				
			|||||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
					    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
				
			||||||
    "unauthorizedButton": "حاول مجددا",
 | 
					    "unauthorizedButton": "حاول مجددا",
 | 
				
			||||||
    "untrustedRedirectTitle": "إعادة توجيه غير موثوقة",
 | 
					    "untrustedRedirectTitle": "إعادة توجيه غير موثوقة",
 | 
				
			||||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
					    "untrustedRedirectSubtitle": "أنت تحاول إعادة التوجيه إلى نطاق لا يتطابق مع النطاق المكون الخاص بك (<code>{{domain}}</code>). هل أنت متأكد من أنك تريد المتابعة؟",
 | 
				
			||||||
    "cancelTitle": "إلغاء",
 | 
					    "cancelTitle": "إلغاء",
 | 
				
			||||||
    "forgotPasswordTitle": "نسيت كلمة المرور؟",
 | 
					    "forgotPasswordTitle": "نسيت كلمة المرور؟",
 | 
				
			||||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
					    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -10,7 +10,7 @@
 | 
				
			|||||||
    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
					    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
				
			||||||
    "loginSuccessTitle": "Connecté",
 | 
					    "loginSuccessTitle": "Connecté",
 | 
				
			||||||
    "loginSuccessSubtitle": "Bienvenue!",
 | 
					    "loginSuccessSubtitle": "Bienvenue!",
 | 
				
			||||||
    "loginOauthFailTitle": "An error occurred",
 | 
					    "loginOauthFailTitle": "Une erreur s'est produite",
 | 
				
			||||||
    "loginOauthFailSubtitle": "Impossible d'obtenir l'URL OAuth",
 | 
					    "loginOauthFailSubtitle": "Impossible d'obtenir l'URL OAuth",
 | 
				
			||||||
    "loginOauthSuccessTitle": "Redirection",
 | 
					    "loginOauthSuccessTitle": "Redirection",
 | 
				
			||||||
    "loginOauthSuccessSubtitle": "Redirection vers votre fournisseur OAuth",
 | 
					    "loginOauthSuccessSubtitle": "Redirection vers votre fournisseur OAuth",
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,54 +1,54 @@
 | 
				
			|||||||
{
 | 
					{
 | 
				
			||||||
    "loginTitle": "Welcome back, login with",
 | 
					    "loginTitle": "歡迎回來,請用以下方式登入",
 | 
				
			||||||
    "loginTitleSimple": "Welcome back, please login",
 | 
					    "loginTitleSimple": "歡迎回來,請登入",
 | 
				
			||||||
    "loginDivider": "Or",
 | 
					    "loginDivider": "或",
 | 
				
			||||||
    "loginUsername": "Username",
 | 
					    "loginUsername": "帳號",
 | 
				
			||||||
    "loginPassword": "Password",
 | 
					    "loginPassword": "密碼",
 | 
				
			||||||
    "loginSubmit": "Login",
 | 
					    "loginSubmit": "登入",
 | 
				
			||||||
    "loginFailTitle": "Failed to log in",
 | 
					    "loginFailTitle": "登入失敗",
 | 
				
			||||||
    "loginFailSubtitle": "Please check your username and password",
 | 
					    "loginFailSubtitle": "請檢查您的帳號與密碼",
 | 
				
			||||||
    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
					    "loginFailRateLimit": "登入失敗次數過多,請稍後再試",
 | 
				
			||||||
    "loginSuccessTitle": "Logged in",
 | 
					    "loginSuccessTitle": "登入成功",
 | 
				
			||||||
    "loginSuccessSubtitle": "Welcome back!",
 | 
					    "loginSuccessSubtitle": "歡迎回來!",
 | 
				
			||||||
    "loginOauthFailTitle": "An error occurred",
 | 
					    "loginOauthFailTitle": "發生錯誤",
 | 
				
			||||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
					    "loginOauthFailSubtitle": "無法取得 OAuth 網址",
 | 
				
			||||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
					    "loginOauthSuccessTitle": "重新導向中",
 | 
				
			||||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
					    "loginOauthSuccessSubtitle": "正在將您重新導向至 OAuth 供應商",
 | 
				
			||||||
    "continueRedirectingTitle": "Redirecting...",
 | 
					    "continueRedirectingTitle": "重新導向中...",
 | 
				
			||||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
					    "continueRedirectingSubtitle": "您即將被重新導向至應用程式",
 | 
				
			||||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
					    "continueInvalidRedirectTitle": "無效的重新導向",
 | 
				
			||||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
					    "continueInvalidRedirectSubtitle": "重新導向的網址無效",
 | 
				
			||||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
					    "continueInsecureRedirectTitle": "不安全的重新導向",
 | 
				
			||||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
					    "continueInsecureRedirectSubtitle": "您正嘗試從安全的 <code>https</code> 重新導向至不安全的 <code>http</code>。您確定要繼續嗎?",
 | 
				
			||||||
    "continueTitle": "Continue",
 | 
					    "continueTitle": "繼續",
 | 
				
			||||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
					    "continueSubtitle": "點擊按鈕以繼續前往您的應用程式。",
 | 
				
			||||||
    "logoutFailTitle": "Failed to log out",
 | 
					    "logoutFailTitle": "登出失敗",
 | 
				
			||||||
    "logoutFailSubtitle": "Please try again",
 | 
					    "logoutFailSubtitle": "請再試一次",
 | 
				
			||||||
    "logoutSuccessTitle": "Logged out",
 | 
					    "logoutSuccessTitle": "登出成功",
 | 
				
			||||||
    "logoutSuccessSubtitle": "You have been logged out",
 | 
					    "logoutSuccessSubtitle": "您已成功登出",
 | 
				
			||||||
    "logoutTitle": "Logout",
 | 
					    "logoutTitle": "登出",
 | 
				
			||||||
    "logoutUsernameSubtitle": "You are currently logged in as <code>{{username}}</code>. Click the button below to logout.",
 | 
					    "logoutUsernameSubtitle": "您目前以 <code>{{username}}</code> 的身分登入。點擊下方按鈕以登出。",
 | 
				
			||||||
    "logoutOauthSubtitle": "You are currently logged in as <code>{{username}}</code> using the {{provider}} OAuth provider. Click the button below to logout.",
 | 
					    "logoutOauthSubtitle": "您目前使用 {{provider}} OAuth 供應商並以 <code>{{username}}</code> 的身分登入。點擊下方按鈕以登出。",
 | 
				
			||||||
    "notFoundTitle": "Page not found",
 | 
					    "notFoundTitle": "找不到頁面",
 | 
				
			||||||
    "notFoundSubtitle": "The page you are looking for does not exist.",
 | 
					    "notFoundSubtitle": "您要尋找的頁面不存在。",
 | 
				
			||||||
    "notFoundButton": "Go home",
 | 
					    "notFoundButton": "回到首頁",
 | 
				
			||||||
    "totpFailTitle": "Failed to verify code",
 | 
					    "totpFailTitle": "驗證失敗",
 | 
				
			||||||
    "totpFailSubtitle": "Please check your code and try again",
 | 
					    "totpFailSubtitle": "請檢查您的驗證碼並再試一次",
 | 
				
			||||||
    "totpSuccessTitle": "Verified",
 | 
					    "totpSuccessTitle": "驗證成功",
 | 
				
			||||||
    "totpSuccessSubtitle": "Redirecting to your app",
 | 
					    "totpSuccessSubtitle": "正在重新導向至您的應用程式",
 | 
				
			||||||
    "totpTitle": "Enter your TOTP code",
 | 
					    "totpTitle": "輸入您的 TOTP 驗證碼",
 | 
				
			||||||
    "totpSubtitle": "Please enter the code from your authenticator app.",
 | 
					    "totpSubtitle": "請輸入您驗證器應用程式中的代碼。",
 | 
				
			||||||
    "unauthorizedTitle": "Unauthorized",
 | 
					    "unauthorizedTitle": "未經授權",
 | 
				
			||||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
					    "unauthorizedResourceSubtitle": "使用者 <code>{{username}}</code> 未被授權存取資源 <code>{{resource}}</code>。",
 | 
				
			||||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
					    "unauthorizedLoginSubtitle": "使用者 <code>{{username}}</code> 未被授權登入。",
 | 
				
			||||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
					    "unauthorizedGroupsSubtitle": "使用者 <code>{{username}}</code> 不在存取資源 <code>{{resource}}</code> 所需的群組中。",
 | 
				
			||||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
					    "unauthorizedIpSubtitle": "您的 IP 位址 <code>{{ip}}</code> 未被授權存取資源 <code>{{resource}}</code>。",
 | 
				
			||||||
    "unauthorizedButton": "Try again",
 | 
					    "unauthorizedButton": "再試一次",
 | 
				
			||||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
					    "untrustedRedirectTitle": "不受信任的重新導向",
 | 
				
			||||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
					    "untrustedRedirectSubtitle": "您正嘗試重新導向至的網域與您設定的網域 (<code>{{domain}}</code>) 不符。您確定要繼續嗎?",
 | 
				
			||||||
    "cancelTitle": "Cancel",
 | 
					    "cancelTitle": "取消",
 | 
				
			||||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
					    "forgotPasswordTitle": "忘記密碼?",
 | 
				
			||||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
					    "failedToFetchProvidersTitle": "載入驗證供應商失敗。請檢查您的設定。",
 | 
				
			||||||
    "errorTitle": "An error occurred",
 | 
					    "errorTitle": "發生錯誤",
 | 
				
			||||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information."
 | 
					    "errorSubtitle": "執行此操作時發生錯誤。請檢查主控台以獲取更多資訊。"
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -234,7 +234,7 @@ func (auth *Auth) RecordLoginAttempt(identifier string, success bool) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func (auth *Auth) EmailWhitelisted(email string) bool {
 | 
					func (auth *Auth) EmailWhitelisted(email string) bool {
 | 
				
			||||||
	return utils.CheckFilter(auth.Config.OauthWhitelist, email, true)
 | 
						return utils.CheckFilter(auth.Config.OauthWhitelist, email)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func (auth *Auth) CreateSessionCookie(c *gin.Context, data *types.SessionCookie) error {
 | 
					func (auth *Auth) CreateSessionCookie(c *gin.Context, data *types.SessionCookie) error {
 | 
				
			||||||
@@ -368,13 +368,13 @@ func (auth *Auth) ResourceAllowed(c *gin.Context, context types.UserContext, lab
 | 
				
			|||||||
	// Check if oauth is allowed
 | 
						// Check if oauth is allowed
 | 
				
			||||||
	if context.OAuth {
 | 
						if context.OAuth {
 | 
				
			||||||
		log.Debug().Msg("Checking OAuth whitelist")
 | 
							log.Debug().Msg("Checking OAuth whitelist")
 | 
				
			||||||
		return utils.CheckFilter(labels.OAuth.Whitelist, context.Email, true)
 | 
							return utils.CheckFilter(labels.OAuth.Whitelist, context.Email)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Check users
 | 
						// Check users
 | 
				
			||||||
	log.Debug().Msg("Checking users")
 | 
						log.Debug().Msg("Checking users")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return utils.CheckFilter(labels.Users, context.Username, true)
 | 
						return utils.CheckFilter(labels.Users, context.Username)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func (auth *Auth) OAuthGroup(c *gin.Context, context types.UserContext, labels types.Labels) bool {
 | 
					func (auth *Auth) OAuthGroup(c *gin.Context, context types.UserContext, labels types.Labels) bool {
 | 
				
			||||||
@@ -394,7 +394,7 @@ func (auth *Auth) OAuthGroup(c *gin.Context, context types.UserContext, labels t
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	// For every group check if it is in the required groups
 | 
						// For every group check if it is in the required groups
 | 
				
			||||||
	for _, group := range oauthGroups {
 | 
						for _, group := range oauthGroups {
 | 
				
			||||||
		if utils.CheckFilter(labels.OAuth.Groups, group, true) {
 | 
							if utils.CheckFilter(labels.OAuth.Groups, group) {
 | 
				
			||||||
			log.Debug().Str("group", group).Msg("Group is in required groups")
 | 
								log.Debug().Str("group", group).Msg("Group is in required groups")
 | 
				
			||||||
			return true
 | 
								return true
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
@@ -452,10 +452,7 @@ func (auth *Auth) GetBasicAuth(c *gin.Context) *types.User {
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func (auth *Auth) CheckIP(c *gin.Context, labels types.Labels) bool {
 | 
					func (auth *Auth) CheckIP(labels types.Labels, ip string) bool {
 | 
				
			||||||
	// Get the IP address from the request
 | 
					 | 
				
			||||||
	ip := c.ClientIP()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	// Check if the IP is in block list
 | 
						// Check if the IP is in block list
 | 
				
			||||||
	for _, blocked := range labels.IP.Block {
 | 
						for _, blocked := range labels.IP.Block {
 | 
				
			||||||
		res, err := utils.FilterIP(blocked, ip)
 | 
							res, err := utils.FilterIP(blocked, ip)
 | 
				
			||||||
@@ -492,3 +489,22 @@ func (auth *Auth) CheckIP(c *gin.Context, labels types.Labels) bool {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	return true
 | 
						return true
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (auth *Auth) BypassedIP(labels types.Labels, ip string) bool {
 | 
				
			||||||
 | 
						// For every IP in the bypass list, check if the IP matches
 | 
				
			||||||
 | 
						for _, bypassed := range labels.IP.Bypass {
 | 
				
			||||||
 | 
							res, err := utils.FilterIP(bypassed, ip)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								log.Warn().Err(err).Str("item", bypassed).Msg("Invalid IP/CIDR in bypass list")
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if res {
 | 
				
			||||||
 | 
								log.Debug().Str("ip", ip).Str("item", bypassed).Msg("IP is in bypass list, allowing access")
 | 
				
			||||||
 | 
								return true
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						log.Debug().Str("ip", ip).Msg("IP not in bypass list, continuing with authentication")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return false
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -69,7 +69,7 @@ func (docker *Docker) DockerConnected() bool {
 | 
				
			|||||||
	return err == nil
 | 
						return err == nil
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func (docker *Docker) GetLabels(id string, domain string) (types.Labels, error) {
 | 
					func (docker *Docker) GetLabels(app string, domain string) (types.Labels, error) {
 | 
				
			||||||
	// Check if we have access to the Docker API
 | 
						// Check if we have access to the Docker API
 | 
				
			||||||
	isConnected := docker.DockerConnected()
 | 
						isConnected := docker.DockerConnected()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -112,9 +112,16 @@ func (docker *Docker) GetLabels(id string, domain string) (types.Labels, error)
 | 
				
			|||||||
			continue
 | 
								continue
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		// Check if the labels match the id or the domain
 | 
							// Check if the container matches the ID or domain
 | 
				
			||||||
		if strings.TrimPrefix(inspect.Name, "/") == id || utils.CheckFilter(labels.Domain, domain, false) { // Disable regex for now
 | 
							for _, lDomain := range labels.Domain {
 | 
				
			||||||
			log.Debug().Str("id", inspect.ID).Msg("Found matching container")
 | 
								if lDomain == domain {
 | 
				
			||||||
 | 
									log.Debug().Str("id", inspect.ID).Msg("Found matching container by domain")
 | 
				
			||||||
 | 
									return labels, nil
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if strings.TrimPrefix(inspect.Name, "/") == app {
 | 
				
			||||||
 | 
								log.Debug().Str("id", inspect.ID).Msg("Found matching container by name")
 | 
				
			||||||
			return labels, nil
 | 
								return labels, nil
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -96,11 +96,29 @@ func (h *Handlers) AuthHandler(c *gin.Context) {
 | 
				
			|||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Check if the IP is allowed/blocked
 | 
						// Get client IP
 | 
				
			||||||
	ip := c.ClientIP()
 | 
						ip := c.ClientIP()
 | 
				
			||||||
	if !h.Auth.CheckIP(c, labels) {
 | 
					 | 
				
			||||||
		log.Warn().Str("ip", ip).Msg("IP not allowed")
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Check if the IP is in bypass list
 | 
				
			||||||
 | 
						if h.Auth.BypassedIP(labels, ip) {
 | 
				
			||||||
 | 
							headersParsed := utils.ParseHeaders(labels.Headers)
 | 
				
			||||||
 | 
							for key, value := range headersParsed {
 | 
				
			||||||
 | 
								log.Debug().Str("key", key).Msg("Setting header")
 | 
				
			||||||
 | 
								c.Header(key, value)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if labels.Basic.Username != "" && utils.GetSecret(labels.Basic.Password.Plain, labels.Basic.Password.File) != "" {
 | 
				
			||||||
 | 
								log.Debug().Str("username", labels.Basic.Username).Msg("Setting basic auth headers")
 | 
				
			||||||
 | 
								c.Header("Authorization", fmt.Sprintf("Basic %s", utils.GetBasicAuth(labels.Basic.Username, utils.GetSecret(labels.Basic.Password.Plain, labels.Basic.Password.File))))
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							c.JSON(200, gin.H{
 | 
				
			||||||
 | 
								"status":  200,
 | 
				
			||||||
 | 
								"message": "Authenticated",
 | 
				
			||||||
 | 
							})
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Check if the IP is allowed/blocked
 | 
				
			||||||
 | 
						if !h.Auth.CheckIP(labels, ip) {
 | 
				
			||||||
		if proxy.Proxy == "nginx" || !isBrowser {
 | 
							if proxy.Proxy == "nginx" || !isBrowser {
 | 
				
			||||||
			c.JSON(403, gin.H{
 | 
								c.JSON(403, gin.H{
 | 
				
			||||||
				"status":  403,
 | 
									"status":  403,
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -122,6 +122,7 @@ type PassowrdLabels struct {
 | 
				
			|||||||
type IPLabels struct {
 | 
					type IPLabels struct {
 | 
				
			||||||
	Allow  []string
 | 
						Allow  []string
 | 
				
			||||||
	Block  []string
 | 
						Block  []string
 | 
				
			||||||
 | 
						Bypass []string
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Labels is a struct that contains the labels for a tinyauth protected container
 | 
					// Labels is a struct that contains the labels for a tinyauth protected container
 | 
				
			||||||
@@ -129,7 +130,7 @@ type Labels struct {
 | 
				
			|||||||
	Users   string
 | 
						Users   string
 | 
				
			||||||
	Allowed string
 | 
						Allowed string
 | 
				
			||||||
	Headers []string
 | 
						Headers []string
 | 
				
			||||||
	Domain  string
 | 
						Domain  []string
 | 
				
			||||||
	Basic   BasicLabels
 | 
						Basic   BasicLabels
 | 
				
			||||||
	OAuth   OAuthLabels
 | 
						OAuth   OAuthLabels
 | 
				
			||||||
	IP      IPLabels
 | 
						IP      IPLabels
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -293,14 +293,14 @@ func ParseSecretFile(contents string) string {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Check if a string matches a regex or if it is included in a comma separated list
 | 
					// Check if a string matches a regex or if it is included in a comma separated list
 | 
				
			||||||
func CheckFilter(filter string, str string, regex bool) bool {
 | 
					func CheckFilter(filter string, str string) bool {
 | 
				
			||||||
	// Check if the filter is empty
 | 
						// Check if the filter is empty
 | 
				
			||||||
	if len(strings.TrimSpace(filter)) == 0 {
 | 
						if len(strings.TrimSpace(filter)) == 0 {
 | 
				
			||||||
		return true
 | 
							return true
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Check if the filter is a regex
 | 
						// Check if the filter is a regex
 | 
				
			||||||
	if strings.HasPrefix(filter, "/") && strings.HasSuffix(filter, "/") && regex {
 | 
						if strings.HasPrefix(filter, "/") && strings.HasSuffix(filter, "/") {
 | 
				
			||||||
		// Create regex
 | 
							// Create regex
 | 
				
			||||||
		re, err := regexp.Compile(filter[1 : len(filter)-1])
 | 
							re, err := regexp.Compile(filter[1 : len(filter)-1])
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -387,7 +387,7 @@ func TestCheckFilter(t *testing.T) {
 | 
				
			|||||||
	expected := true
 | 
						expected := true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Test the check filter function
 | 
						// Test the check filter function
 | 
				
			||||||
	result := utils.CheckFilter(filter, str, false)
 | 
						result := utils.CheckFilter(filter, str)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Check if the result is equal to the expected
 | 
						// Check if the result is equal to the expected
 | 
				
			||||||
	if result != expected {
 | 
						if result != expected {
 | 
				
			||||||
@@ -402,7 +402,7 @@ func TestCheckFilter(t *testing.T) {
 | 
				
			|||||||
	expected = true
 | 
						expected = true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Test the check filter function
 | 
						// Test the check filter function
 | 
				
			||||||
	result = utils.CheckFilter(filter, str, true)
 | 
						result = utils.CheckFilter(filter, str)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Check if the result is equal to the expected
 | 
						// Check if the result is equal to the expected
 | 
				
			||||||
	if result != expected {
 | 
						if result != expected {
 | 
				
			||||||
@@ -417,7 +417,7 @@ func TestCheckFilter(t *testing.T) {
 | 
				
			|||||||
	expected = true
 | 
						expected = true
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Test the check filter function
 | 
						// Test the check filter function
 | 
				
			||||||
	result = utils.CheckFilter(filter, str, false)
 | 
						result = utils.CheckFilter(filter, str)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Check if the result is equal to the expected
 | 
						// Check if the result is equal to the expected
 | 
				
			||||||
	if result != expected {
 | 
						if result != expected {
 | 
				
			||||||
@@ -432,7 +432,7 @@ func TestCheckFilter(t *testing.T) {
 | 
				
			|||||||
	expected = false
 | 
						expected = false
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Test the check filter function
 | 
						// Test the check filter function
 | 
				
			||||||
	result = utils.CheckFilter(filter, str, true)
 | 
						result = utils.CheckFilter(filter, str)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Check if the result is equal to the expected
 | 
						// Check if the result is equal to the expected
 | 
				
			||||||
	if result != expected {
 | 
						if result != expected {
 | 
				
			||||||
@@ -447,7 +447,7 @@ func TestCheckFilter(t *testing.T) {
 | 
				
			|||||||
	expected = false
 | 
						expected = false
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Test the check filter function
 | 
						// Test the check filter function
 | 
				
			||||||
	result = utils.CheckFilter(filter, str, false)
 | 
						result = utils.CheckFilter(filter, str)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Check if the result is equal to the expected
 | 
						// Check if the result is equal to the expected
 | 
				
			||||||
	if result != expected {
 | 
						if result != expected {
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user