Consolidated
Some checks are pending
/ release (push) Waiting to run
testing / backend-checks (push) Waiting to run
testing / frontend-checks (push) Waiting to run
testing / test-unit (push) Blocked by required conditions
testing / test-e2e (push) Blocked by required conditions
testing / test-remote-cacher (redis) (push) Blocked by required conditions
testing / test-remote-cacher (valkey) (push) Blocked by required conditions
testing / test-remote-cacher (garnet) (push) Blocked by required conditions
testing / test-remote-cacher (redict) (push) Blocked by required conditions
testing / test-mysql (push) Blocked by required conditions
testing / test-pgsql (push) Blocked by required conditions
testing / test-sqlite (push) Blocked by required conditions
testing / security-check (push) Blocked by required conditions

Signed-off-by: Minecon724 <minecon724@noreply.git.m724.eu>
This commit is contained in:
Minecon724 2025-03-20 19:34:59 +01:00
parent c1e9fd738b
commit 30c7598c02
18 changed files with 91 additions and 11 deletions

25
ABOUT-FORK.md Normal file
View file

@ -0,0 +1,25 @@
## FORK
This is a fork of Forgejo.
This fork doesn't add any specialized features. It adds things that should be in Forgejo, but aren't.
This fork is based on a release. There's no master branch. Patches of this fork do not follow releases; because of scope, they're released immediately.
This fork is a drop-in replacement. You can switch back and forth, as there are no breaking changes.
### Features
- Privacy Policy support
- Consent checkbox
- Footer link
- Configured with `service.PRIVACY_POLICY_URL`
- Captcha tweaks
- Shorter: 4-5 instead of 6 digits
- Bigger
- Fixed colors on dark theme
- Aesthetics
### Builds
Docker builds [are available here](https://hub.docker.com/r/minecon724/forgejo724) for amd64 and arm64.

View file

@ -1,3 +1,9 @@
## FORK
This is a fork of Forgejo. [Read more here.](ABOUT-FORK.md)
<div align="center"> <div align="center">
<img src="./assets/logo.svg" alt="" width="192" align="center" /> <img src="./assets/logo.svg" alt="" width="192" align="center" />
<h1 align="center">Welcome to Forgejo</h1> <h1 align="center">Welcome to Forgejo</h1>

View file

@ -85,6 +85,7 @@ var Service = struct {
DefaultOrgMemberVisible bool DefaultOrgMemberVisible bool
UserDeleteWithCommentsMaxTime time.Duration UserDeleteWithCommentsMaxTime time.Duration
ValidSiteURLSchemes []string ValidSiteURLSchemes []string
PrivacyPolicyURL string
// OpenID settings // OpenID settings
EnableOpenIDSignIn bool EnableOpenIDSignIn bool
@ -263,6 +264,8 @@ func loadServiceFrom(rootCfg ConfigProvider) {
} }
Service.ValidSiteURLSchemes = schemes Service.ValidSiteURLSchemes = schemes
Service.PrivacyPolicyURL = sec.Key("PRIVACY_POLICY_URL").MustString("")
mustMapSetting(rootCfg, "service.explore", &Service.Explore) mustMapSetting(rootCfg, "service.explore", &Service.Explore)
loadOpenIDSetting(rootCfg) loadOpenIDSetting(rootCfg)

View file

@ -58,6 +58,7 @@ func CommonTemplateContextData() ContextData {
"EnableOpenIDSignIn": setting.Service.EnableOpenIDSignIn, "EnableOpenIDSignIn": setting.Service.EnableOpenIDSignIn,
"PageStartTime": time.Now(), "PageStartTime": time.Now(),
"RunModeIsProd": setting.IsProd, "RunModeIsProd": setting.IsProd,
"PrivacyPolicyURL": setting.Service.PrivacyPolicyURL,
} }
} }

View file

@ -28,6 +28,7 @@ licenses = Licenses
return_to_forgejo = Return to Forgejo return_to_forgejo = Return to Forgejo
toggle_menu = Toggle menu toggle_menu = Toggle menu
more_items = More items more_items = More items
privacy_policy = Privacy Policy
username = Username username = Username
email = Email address email = Email address
@ -38,6 +39,7 @@ captcha = CAPTCHA
twofa = Two-factor authentication twofa = Two-factor authentication
twofa_scratch = Two-factor scratch code twofa_scratch = Two-factor scratch code
passcode = Passcode passcode = Passcode
consent_agree = I agree to the <a href="%s">Privacy Policy</a>
webauthn_insert_key = Insert your security key webauthn_insert_key = Insert your security key
webauthn_sign_in = Press the button on your security key. If your security key has no button, re-insert it. webauthn_sign_in = Press the button on your security key. If your security key has no button, re-insert it.
@ -485,6 +487,7 @@ password_pwned_err = Could not complete request to HaveIBeenPwned
last_admin = You cannot remove the last admin. There must be at least one admin. last_admin = You cannot remove the last admin. There must be at least one admin.
back_to_sign_in = Back to Sign in back_to_sign_in = Back to Sign in
sign_in_openid = Proceed with OpenID sign_in_openid = Proceed with OpenID
must_consent = Agreement to our Privacy Policy is required to register.
[mail] [mail]
view_it_on = View it on %s view_it_on = View it on %s

View file

@ -482,6 +482,12 @@ func SignUpPost(ctx *context.Context) {
ctx.RenderWithErr(password.BuildComplexityError(ctx.Locale), tplSignUp, &form) ctx.RenderWithErr(password.BuildComplexityError(ctx.Locale), tplSignUp, &form)
return return
} }
if !form.Consent {
ctx.RenderWithErr(ctx.Tr("auth.must_consent"), tplSignUp, &form)
return
} // consent is required before sending password anywhere
if err := password.IsPwned(ctx, form.Password); err != nil { if err := password.IsPwned(ctx, form.Password); err != nil {
errMsg := ctx.Tr("auth.password_pwned", "https://haveibeenpwned.com/Passwords") errMsg := ctx.Tr("auth.password_pwned", "https://haveibeenpwned.com/Passwords")
if password.IsErrIsPwnedRequest(err) { if password.IsErrIsPwnedRequest(err) {

View file

@ -5,6 +5,7 @@ package context
import ( import (
"fmt" "fmt"
"math/rand"
"sync" "sync"
"code.gitea.io/gitea/modules/base" "code.gitea.io/gitea/modules/base"
@ -55,7 +56,9 @@ func GetImageCaptcha() string {
imageCaptchaOnce.Do(func() { imageCaptchaOnce.Do(func() {
captcha.SetCustomStore(&imageCaptchaStore{c: cache.GetCache()}) captcha.SetCustomStore(&imageCaptchaStore{c: cache.GetCache()})
}) })
return captcha.New()
length := 4 + rand.Intn(2) // 4 or 5 chars
return captcha.NewLen(length)
} }
// SetCaptchaData sets common captcha data // SetCaptchaData sets common captcha data

View file

@ -96,6 +96,7 @@ type RegisterForm struct {
Email string `binding:"Required;MaxSize(254)"` Email string `binding:"Required;MaxSize(254)"`
Password string `binding:"MaxSize(255)"` Password string `binding:"MaxSize(255)"`
Retype string Retype string
Consent bool
} }
// Validate validates the fields // Validate validates the fields

View file

@ -4,11 +4,10 @@
<a target="_blank" rel="noopener noreferrer" href="https://forgejo.org">{{ctx.Locale.Tr "powered_by" "Forgejo"}}</a> <a target="_blank" rel="noopener noreferrer" href="https://forgejo.org">{{ctx.Locale.Tr "powered_by" "Forgejo"}}</a>
{{end}} {{end}}
{{if (or .ShowFooterVersion .PageIsAdmin)}} {{if (or .ShowFooterVersion .PageIsAdmin)}}
{{ctx.Locale.Tr "version"}}:
{{if .IsAdmin}} {{if .IsAdmin}}
<a href="{{AppSubUrl}}/admin/config">{{AppVer}}</a> <a href="{{AppSubUrl}}/admin/config">v{{AppVer}}</a>
{{else}} {{else}}
{{AppVerNoMetadata}} v{{AppVerNoMetadata}}
{{end}} {{end}}
{{end}} {{end}}
{{if and .TemplateLoadTimes ShowFooterTemplateLoadTime}} {{if and .TemplateLoadTimes ShowFooterTemplateLoadTime}}
@ -26,7 +25,7 @@
</div> </div>
</div> </div>
<a href="{{AssetUrlPrefix}}/licenses.txt">{{ctx.Locale.Tr "licenses"}}</a> <a href="{{AssetUrlPrefix}}/licenses.txt">{{ctx.Locale.Tr "licenses"}}</a>
{{if .EnableSwagger}}<a href="{{AppSubUrl}}/api/swagger">API</a>{{end}} {{if .PrivacyPolicyURL}}<a href="{{.PrivacyPolicyURL}}">{{ctx.Locale.Tr "privacy_policy"}}</a>{{end}}
{{template "custom/extra_links_footer" .}} {{template "custom/extra_links_footer" .}}
</div> </div>
</footer> </footer>

View file

@ -37,6 +37,17 @@
{{template "user/auth/captcha" .}} {{template "user/auth/captcha" .}}
{{ if (.PrivacyPolicyURL) }}
<div class="inline field required">
<div class="ui checkbox">
<input id="consent" name="consent" type="checkbox">
<label for="consent">{{ ctx.Locale.Tr "consent_agree" .PrivacyPolicyURL }}</label>
</div>
</div>
{{ else }}
<input id="consent" name="consent" type="checkbox" checked hidden>
{{ end }}
<div class="inline field"> <div class="inline field">
<button class="ui primary button tw-w-full"> <button class="ui primary button tw-w-full">
{{if .LinkAccountMode}} {{if .LinkAccountMode}}

View file

@ -372,6 +372,7 @@ a.label,
color: var(--color-text); color: var(--color-text);
user-select: auto; user-select: auto;
line-height: var(--line-height-default); /* fomantic uses "1" which causes overflow problems because "1" doesn't consider the descent part */ line-height: var(--line-height-default); /* fomantic uses "1" which causes overflow problems because "1" doesn't consider the descent part */
transition: none;
} }
.ui.menu .item > .svg { .ui.menu .item > .svg {

View file

@ -47,9 +47,16 @@ fieldset label + .ui.dropdown {
fieldset label > input[type="checkbox"], fieldset label > input[type="checkbox"],
fieldset label > input[type="radio"] { fieldset label > input[type="radio"] {
margin-right: 0.75em; margin-right: 0.5em;
margin-top: 0 !important; margin-top: 0 !important;
vertical-align: initial !important; /* overrides a semantic.css rule, remove when obsolete */ vertical-align: initial !important; /* overrides a semantic.css rule, remove when obsolete */
cursor: pointer;
}
fieldset label:has(input[type="checkbox"]),
fieldset label:has(input[type="radio"]) {
cursor: pointer;
} }
@media (min-width: 768px) { @media (min-width: 768px) {
@ -334,6 +341,11 @@ input:-webkit-autofill:active,
display: inline-block; display: inline-block;
} }
.captcha-img {
width: min(30em, 100%);
}
@media (min-width: 768px) { @media (min-width: 768px) {
.g-recaptcha-style, .g-recaptcha-style,
.h-captcha-style { .h-captcha-style {

View file

@ -70,7 +70,7 @@
.page-footer .right-links > a { .page-footer .right-links > a {
border-left: 1px solid var(--color-secondary-dark-1); border-left: 1px solid var(--color-secondary-dark-1);
padding-left: 8px; padding-left: 8px;
margin-left: 5px; margin-left: 8px;
} }
.page-footer .ui.dropdown.language .menu { .page-footer .ui.dropdown.language .menu {

View file

@ -11,3 +11,10 @@
.markup [href$="#dark-mode-only"] { .markup [href$="#dark-mode-only"] {
display: unset; display: unset;
} }
/* The reason we're doing this *here* is because dark vanilla templates always use this file, and there's no other way to detect dark theme */
.captcha-img {
filter: invert();
}

View file

@ -5,6 +5,7 @@
background: var(--color-button); background: var(--color-button);
border: 1px solid var(--color-light-border); border: 1px solid var(--color-light-border);
color: var(--color-text); color: var(--color-text);
transition: none;
} }
.ui.button:hover, .ui.button:hover,

View file

@ -41,7 +41,8 @@ input[type="radio"] {
.ui.checkbox label, .ui.checkbox label,
.ui.radio.checkbox label { .ui.radio.checkbox label {
margin-left: 1.85714em; padding-left: 1.85714em;
cursor: pointer;
} }
.ui.checkbox + label { .ui.checkbox + label {

View file

@ -231,7 +231,7 @@
--color-markup-table-row: #ffffff06; --color-markup-table-row: #ffffff06;
--color-markup-code-block: var(--steel-800); --color-markup-code-block: var(--steel-800);
--color-markup-code-inline: var(--steel-850); --color-markup-code-inline: var(--steel-850);
--color-button: var(--steel-600); --color-button: var(--steel-650);
--color-code-bg: var(--steel-750); --color-code-bg: var(--steel-750);
--color-shadow: #00000060; --color-shadow: #00000060;
--color-secondary-bg: var(--steel-700); --color-secondary-bg: var(--steel-700);

View file

@ -247,7 +247,7 @@
--color-markup-table-row: #ffffff06; --color-markup-table-row: #ffffff06;
--color-markup-code-block: var(--zinc-150); --color-markup-code-block: var(--zinc-150);
--color-markup-code-inline: var(--zinc-200); --color-markup-code-inline: var(--zinc-200);
--color-button: var(--zinc-150); --color-button: var(--zinc-100);
--color-code-bg: var(--zinc-50); --color-code-bg: var(--zinc-50);
--color-shadow: #00000060; --color-shadow: #00000060;
--color-secondary-bg: var(--zinc-100); --color-secondary-bg: var(--zinc-100);