mirror of
https://github.com/go-gitea/gitea
synced 2024-12-30 18:45:59 +01:00
Merge branch 'main' into actions-auto-expand-running-step
This commit is contained in:
commit
bb0a7d0954
@ -2,9 +2,10 @@ root = "."
|
|||||||
tmp_dir = ".air"
|
tmp_dir = ".air"
|
||||||
|
|
||||||
[build]
|
[build]
|
||||||
|
pre_cmd = ["killall -9 gitea 2>/dev/null || true"] # kill off potential zombie processes from previous runs
|
||||||
cmd = "make --no-print-directory backend"
|
cmd = "make --no-print-directory backend"
|
||||||
bin = "gitea"
|
bin = "gitea"
|
||||||
delay = 1000
|
delay = 2000
|
||||||
include_ext = ["go", "tmpl"]
|
include_ext = ["go", "tmpl"]
|
||||||
include_file = ["main.go"]
|
include_file = ["main.go"]
|
||||||
include_dir = ["cmd", "models", "modules", "options", "routers", "services"]
|
include_dir = ["cmd", "models", "modules", "options", "routers", "services"]
|
||||||
@ -20,3 +21,6 @@ exclude_dir = [
|
|||||||
]
|
]
|
||||||
exclude_regex = ["_test.go$", "_gen.go$"]
|
exclude_regex = ["_test.go$", "_gen.go$"]
|
||||||
stop_on_error = true
|
stop_on_error = true
|
||||||
|
|
||||||
|
[log]
|
||||||
|
main_only = true
|
||||||
|
@ -10,7 +10,8 @@
|
|||||||
"ghcr.io/devcontainers-contrib/features/poetry:2": {},
|
"ghcr.io/devcontainers-contrib/features/poetry:2": {},
|
||||||
"ghcr.io/devcontainers/features/python:1": {
|
"ghcr.io/devcontainers/features/python:1": {
|
||||||
"version": "3.12"
|
"version": "3.12"
|
||||||
}
|
},
|
||||||
|
"ghcr.io/warrenbuckley/codespace-features/sqlite:1": {}
|
||||||
},
|
},
|
||||||
"customizations": {
|
"customizations": {
|
||||||
"vscode": {
|
"vscode": {
|
||||||
@ -25,8 +26,9 @@
|
|||||||
"Vue.volar",
|
"Vue.volar",
|
||||||
"ms-azuretools.vscode-docker",
|
"ms-azuretools.vscode-docker",
|
||||||
"vitest.explorer",
|
"vitest.explorer",
|
||||||
"qwtel.sqlite-viewer",
|
"cweijan.vscode-database-client2",
|
||||||
"GitHub.vscode-pull-request-github"
|
"GitHub.vscode-pull-request-github",
|
||||||
|
"Azurite.azurite"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -14,7 +14,7 @@ _test
|
|||||||
|
|
||||||
# MS VSCode
|
# MS VSCode
|
||||||
.vscode
|
.vscode
|
||||||
__debug_bin
|
__debug_bin*
|
||||||
|
|
||||||
# Architecture specific extensions/prefixes
|
# Architecture specific extensions/prefixes
|
||||||
*.[568vq]
|
*.[568vq]
|
||||||
@ -78,7 +78,6 @@ cpu.out
|
|||||||
/public/assets/css
|
/public/assets/css
|
||||||
/public/assets/fonts
|
/public/assets/fonts
|
||||||
/public/assets/img/avatar
|
/public/assets/img/avatar
|
||||||
/public/assets/img/webpack
|
|
||||||
/vendor
|
/vendor
|
||||||
/web_src/fomantic/node_modules
|
/web_src/fomantic/node_modules
|
||||||
/web_src/fomantic/build/*
|
/web_src/fomantic/build/*
|
||||||
@ -96,6 +95,9 @@ cpu.out
|
|||||||
/.air
|
/.air
|
||||||
/.go-licenses
|
/.go-licenses
|
||||||
|
|
||||||
|
# Files and folders that were previously generated
|
||||||
|
/public/assets/img/webpack
|
||||||
|
|
||||||
# Snapcraft
|
# Snapcraft
|
||||||
snap/.snapcraft/
|
snap/.snapcraft/
|
||||||
parts/
|
parts/
|
||||||
|
264
.eslintrc.yaml
264
.eslintrc.yaml
@ -3,18 +3,33 @@ reportUnusedDisableDirectives: true
|
|||||||
|
|
||||||
ignorePatterns:
|
ignorePatterns:
|
||||||
- /web_src/js/vendor
|
- /web_src/js/vendor
|
||||||
|
- /web_src/fomantic
|
||||||
|
- /public/assets/js
|
||||||
|
|
||||||
|
parser: "@typescript-eslint/parser"
|
||||||
|
|
||||||
parserOptions:
|
parserOptions:
|
||||||
sourceType: module
|
sourceType: module
|
||||||
ecmaVersion: latest
|
ecmaVersion: latest
|
||||||
|
project: true
|
||||||
|
extraFileExtensions: [".vue"]
|
||||||
|
parser: "@typescript-eslint/parser" # for vue plugin - https://eslint.vuejs.org/user-guide/#how-to-use-a-custom-parser
|
||||||
|
|
||||||
|
settings:
|
||||||
|
import/extensions: [".js", ".ts"]
|
||||||
|
import/parsers:
|
||||||
|
"@typescript-eslint/parser": [".js", ".ts"]
|
||||||
|
import/resolver:
|
||||||
|
typescript: true
|
||||||
|
|
||||||
plugins:
|
plugins:
|
||||||
- "@eslint-community/eslint-plugin-eslint-comments"
|
- "@eslint-community/eslint-plugin-eslint-comments"
|
||||||
- "@stylistic/eslint-plugin-js"
|
- "@stylistic/eslint-plugin-js"
|
||||||
|
- "@typescript-eslint/eslint-plugin"
|
||||||
- eslint-plugin-array-func
|
- eslint-plugin-array-func
|
||||||
|
- eslint-plugin-deprecation
|
||||||
- eslint-plugin-github
|
- eslint-plugin-github
|
||||||
- eslint-plugin-i
|
- eslint-plugin-i
|
||||||
- eslint-plugin-jquery
|
|
||||||
- eslint-plugin-no-jquery
|
- eslint-plugin-no-jquery
|
||||||
- eslint-plugin-no-use-extend-native
|
- eslint-plugin-no-use-extend-native
|
||||||
- eslint-plugin-regexp
|
- eslint-plugin-regexp
|
||||||
@ -45,7 +60,15 @@ overrides:
|
|||||||
- files: ["*.config.*"]
|
- files: ["*.config.*"]
|
||||||
rules:
|
rules:
|
||||||
i/no-unused-modules: [0]
|
i/no-unused-modules: [0]
|
||||||
- files: ["**/*.test.*", "web_src/js/test/setup.js"]
|
- files: ["**/*.d.ts"]
|
||||||
|
rules:
|
||||||
|
i/no-unused-modules: [0]
|
||||||
|
"@typescript-eslint/consistent-type-definitions": [0]
|
||||||
|
"@typescript-eslint/consistent-type-imports": [0]
|
||||||
|
- files: ["web_src/js/types.ts"]
|
||||||
|
rules:
|
||||||
|
i/no-unused-modules: [0]
|
||||||
|
- files: ["**/*.test.*", "web_src/js/test/setup.ts"]
|
||||||
env:
|
env:
|
||||||
vitest-globals/env: true
|
vitest-globals/env: true
|
||||||
rules:
|
rules:
|
||||||
@ -98,9 +121,25 @@ overrides:
|
|||||||
vitest/valid-describe-callback: [2]
|
vitest/valid-describe-callback: [2]
|
||||||
vitest/valid-expect: [2]
|
vitest/valid-expect: [2]
|
||||||
vitest/valid-title: [2]
|
vitest/valid-title: [2]
|
||||||
- files: ["web_src/js/modules/fetch.js", "web_src/js/standalone/**/*"]
|
- files: ["web_src/js/modules/fetch.ts", "web_src/js/standalone/**/*"]
|
||||||
rules:
|
rules:
|
||||||
no-restricted-syntax: [2, WithStatement, ForInStatement, LabeledStatement, SequenceExpression]
|
no-restricted-syntax: [2, WithStatement, ForInStatement, LabeledStatement, SequenceExpression]
|
||||||
|
- files: ["**/*.vue"]
|
||||||
|
plugins:
|
||||||
|
- eslint-plugin-vue
|
||||||
|
- eslint-plugin-vue-scoped-css
|
||||||
|
extends:
|
||||||
|
- plugin:vue/vue3-recommended
|
||||||
|
- plugin:vue-scoped-css/vue3-recommended
|
||||||
|
rules:
|
||||||
|
vue/attributes-order: [0]
|
||||||
|
vue/html-closing-bracket-spacing: [2, {startTag: never, endTag: never, selfClosingTag: never}]
|
||||||
|
vue/max-attributes-per-line: [0]
|
||||||
|
vue/singleline-html-element-content-newline: [0]
|
||||||
|
- files: ["tests/e2e/**"]
|
||||||
|
plugins:
|
||||||
|
- eslint-plugin-playwright
|
||||||
|
extends: plugin:playwright/recommended
|
||||||
|
|
||||||
rules:
|
rules:
|
||||||
"@eslint-community/eslint-comments/disable-enable-pair": [2]
|
"@eslint-community/eslint-comments/disable-enable-pair": [2]
|
||||||
@ -125,19 +164,21 @@ rules:
|
|||||||
"@stylistic/js/computed-property-spacing": [2, never]
|
"@stylistic/js/computed-property-spacing": [2, never]
|
||||||
"@stylistic/js/dot-location": [2, property]
|
"@stylistic/js/dot-location": [2, property]
|
||||||
"@stylistic/js/eol-last": [2]
|
"@stylistic/js/eol-last": [2]
|
||||||
"@stylistic/js/function-call-spacing": [2, never]
|
|
||||||
"@stylistic/js/function-call-argument-newline": [0]
|
"@stylistic/js/function-call-argument-newline": [0]
|
||||||
|
"@stylistic/js/function-call-spacing": [2, never]
|
||||||
"@stylistic/js/function-paren-newline": [0]
|
"@stylistic/js/function-paren-newline": [0]
|
||||||
"@stylistic/js/generator-star-spacing": [0]
|
"@stylistic/js/generator-star-spacing": [0]
|
||||||
"@stylistic/js/implicit-arrow-linebreak": [0]
|
"@stylistic/js/implicit-arrow-linebreak": [0]
|
||||||
"@stylistic/js/indent": [2, 2, {ignoreComments: true, SwitchCase: 1}]
|
"@stylistic/js/indent": [2, 2, {ignoreComments: true, SwitchCase: 1}]
|
||||||
"@stylistic/js/key-spacing": [2]
|
"@stylistic/js/key-spacing": [2]
|
||||||
"@stylistic/js/keyword-spacing": [2]
|
"@stylistic/js/keyword-spacing": [2]
|
||||||
|
"@stylistic/js/line-comment-position": [0]
|
||||||
"@stylistic/js/linebreak-style": [2, unix]
|
"@stylistic/js/linebreak-style": [2, unix]
|
||||||
"@stylistic/js/lines-around-comment": [0]
|
"@stylistic/js/lines-around-comment": [0]
|
||||||
"@stylistic/js/lines-between-class-members": [0]
|
"@stylistic/js/lines-between-class-members": [0]
|
||||||
"@stylistic/js/max-len": [0]
|
"@stylistic/js/max-len": [0]
|
||||||
"@stylistic/js/max-statements-per-line": [0]
|
"@stylistic/js/max-statements-per-line": [0]
|
||||||
|
"@stylistic/js/multiline-comment-style": [0]
|
||||||
"@stylistic/js/multiline-ternary": [0]
|
"@stylistic/js/multiline-ternary": [0]
|
||||||
"@stylistic/js/new-parens": [2]
|
"@stylistic/js/new-parens": [2]
|
||||||
"@stylistic/js/newline-per-chained-call": [0]
|
"@stylistic/js/newline-per-chained-call": [0]
|
||||||
@ -167,7 +208,7 @@ rules:
|
|||||||
"@stylistic/js/semi-spacing": [2, {before: false, after: true}]
|
"@stylistic/js/semi-spacing": [2, {before: false, after: true}]
|
||||||
"@stylistic/js/semi-style": [2, last]
|
"@stylistic/js/semi-style": [2, last]
|
||||||
"@stylistic/js/space-before-blocks": [2, always]
|
"@stylistic/js/space-before-blocks": [2, always]
|
||||||
"@stylistic/js/space-before-function-paren": [0]
|
"@stylistic/js/space-before-function-paren": [2, {anonymous: ignore, named: never, asyncArrow: always}]
|
||||||
"@stylistic/js/space-in-parens": [2, never]
|
"@stylistic/js/space-in-parens": [2, never]
|
||||||
"@stylistic/js/space-infix-ops": [2]
|
"@stylistic/js/space-infix-ops": [2]
|
||||||
"@stylistic/js/space-unary-ops": [2]
|
"@stylistic/js/space-unary-ops": [2]
|
||||||
@ -178,6 +219,125 @@ rules:
|
|||||||
"@stylistic/js/wrap-iife": [2, inside]
|
"@stylistic/js/wrap-iife": [2, inside]
|
||||||
"@stylistic/js/wrap-regex": [0]
|
"@stylistic/js/wrap-regex": [0]
|
||||||
"@stylistic/js/yield-star-spacing": [2, after]
|
"@stylistic/js/yield-star-spacing": [2, after]
|
||||||
|
"@typescript-eslint/adjacent-overload-signatures": [0]
|
||||||
|
"@typescript-eslint/array-type": [0]
|
||||||
|
"@typescript-eslint/await-thenable": [2]
|
||||||
|
"@typescript-eslint/ban-ts-comment": [2, {'ts-expect-error': false, 'ts-ignore': true, 'ts-nocheck': false, 'ts-check': false}]
|
||||||
|
"@typescript-eslint/ban-tslint-comment": [0]
|
||||||
|
"@typescript-eslint/class-literal-property-style": [0]
|
||||||
|
"@typescript-eslint/class-methods-use-this": [0]
|
||||||
|
"@typescript-eslint/consistent-generic-constructors": [0]
|
||||||
|
"@typescript-eslint/consistent-indexed-object-style": [0]
|
||||||
|
"@typescript-eslint/consistent-return": [0]
|
||||||
|
"@typescript-eslint/consistent-type-assertions": [2, {assertionStyle: as, objectLiteralTypeAssertions: allow}]
|
||||||
|
"@typescript-eslint/consistent-type-definitions": [2, type]
|
||||||
|
"@typescript-eslint/consistent-type-exports": [2, {fixMixedExportsWithInlineTypeSpecifier: false}]
|
||||||
|
"@typescript-eslint/consistent-type-imports": [2, {prefer: type-imports, fixStyle: separate-type-imports, disallowTypeAnnotations: true}]
|
||||||
|
"@typescript-eslint/default-param-last": [0]
|
||||||
|
"@typescript-eslint/dot-notation": [0]
|
||||||
|
"@typescript-eslint/explicit-function-return-type": [0]
|
||||||
|
"@typescript-eslint/explicit-member-accessibility": [0]
|
||||||
|
"@typescript-eslint/explicit-module-boundary-types": [0]
|
||||||
|
"@typescript-eslint/init-declarations": [0]
|
||||||
|
"@typescript-eslint/max-params": [0]
|
||||||
|
"@typescript-eslint/member-ordering": [0]
|
||||||
|
"@typescript-eslint/method-signature-style": [0]
|
||||||
|
"@typescript-eslint/naming-convention": [0]
|
||||||
|
"@typescript-eslint/no-array-constructor": [2]
|
||||||
|
"@typescript-eslint/no-array-delete": [2]
|
||||||
|
"@typescript-eslint/no-base-to-string": [0]
|
||||||
|
"@typescript-eslint/no-confusing-non-null-assertion": [2]
|
||||||
|
"@typescript-eslint/no-confusing-void-expression": [0]
|
||||||
|
"@typescript-eslint/no-dupe-class-members": [0]
|
||||||
|
"@typescript-eslint/no-duplicate-enum-values": [2]
|
||||||
|
"@typescript-eslint/no-duplicate-type-constituents": [2, {ignoreUnions: true}]
|
||||||
|
"@typescript-eslint/no-dynamic-delete": [0]
|
||||||
|
"@typescript-eslint/no-empty-function": [0]
|
||||||
|
"@typescript-eslint/no-empty-interface": [0]
|
||||||
|
"@typescript-eslint/no-empty-object-type": [2]
|
||||||
|
"@typescript-eslint/no-explicit-any": [0]
|
||||||
|
"@typescript-eslint/no-extra-non-null-assertion": [2]
|
||||||
|
"@typescript-eslint/no-extraneous-class": [0]
|
||||||
|
"@typescript-eslint/no-floating-promises": [0]
|
||||||
|
"@typescript-eslint/no-for-in-array": [2]
|
||||||
|
"@typescript-eslint/no-implied-eval": [2]
|
||||||
|
"@typescript-eslint/no-import-type-side-effects": [0] # dupe with consistent-type-imports
|
||||||
|
"@typescript-eslint/no-inferrable-types": [0]
|
||||||
|
"@typescript-eslint/no-invalid-this": [0]
|
||||||
|
"@typescript-eslint/no-invalid-void-type": [0]
|
||||||
|
"@typescript-eslint/no-loop-func": [0]
|
||||||
|
"@typescript-eslint/no-loss-of-precision": [0]
|
||||||
|
"@typescript-eslint/no-magic-numbers": [0]
|
||||||
|
"@typescript-eslint/no-meaningless-void-operator": [0]
|
||||||
|
"@typescript-eslint/no-misused-new": [2]
|
||||||
|
"@typescript-eslint/no-misused-promises": [2, {checksVoidReturn: {attributes: false, arguments: false}}]
|
||||||
|
"@typescript-eslint/no-mixed-enums": [0]
|
||||||
|
"@typescript-eslint/no-namespace": [2]
|
||||||
|
"@typescript-eslint/no-non-null-asserted-nullish-coalescing": [0]
|
||||||
|
"@typescript-eslint/no-non-null-asserted-optional-chain": [2]
|
||||||
|
"@typescript-eslint/no-non-null-assertion": [0]
|
||||||
|
"@typescript-eslint/no-redeclare": [0]
|
||||||
|
"@typescript-eslint/no-redundant-type-constituents": [2]
|
||||||
|
"@typescript-eslint/no-require-imports": [2]
|
||||||
|
"@typescript-eslint/no-restricted-imports": [0]
|
||||||
|
"@typescript-eslint/no-restricted-types": [0]
|
||||||
|
"@typescript-eslint/no-shadow": [0]
|
||||||
|
"@typescript-eslint/no-this-alias": [0] # handled by unicorn/no-this-assignment
|
||||||
|
"@typescript-eslint/no-unnecessary-boolean-literal-compare": [0]
|
||||||
|
"@typescript-eslint/no-unnecessary-condition": [0]
|
||||||
|
"@typescript-eslint/no-unnecessary-qualifier": [0]
|
||||||
|
"@typescript-eslint/no-unnecessary-template-expression": [0]
|
||||||
|
"@typescript-eslint/no-unnecessary-type-arguments": [0]
|
||||||
|
"@typescript-eslint/no-unnecessary-type-assertion": [2]
|
||||||
|
"@typescript-eslint/no-unnecessary-type-constraint": [2]
|
||||||
|
"@typescript-eslint/no-unsafe-argument": [0]
|
||||||
|
"@typescript-eslint/no-unsafe-assignment": [0]
|
||||||
|
"@typescript-eslint/no-unsafe-call": [0]
|
||||||
|
"@typescript-eslint/no-unsafe-declaration-merging": [2]
|
||||||
|
"@typescript-eslint/no-unsafe-enum-comparison": [2]
|
||||||
|
"@typescript-eslint/no-unsafe-function-type": [2]
|
||||||
|
"@typescript-eslint/no-unsafe-member-access": [0]
|
||||||
|
"@typescript-eslint/no-unsafe-return": [0]
|
||||||
|
"@typescript-eslint/no-unsafe-unary-minus": [2]
|
||||||
|
"@typescript-eslint/no-unused-expressions": [0]
|
||||||
|
"@typescript-eslint/no-unused-vars": [2, {vars: all, args: all, caughtErrors: all, ignoreRestSiblings: false, argsIgnorePattern: ^_, varsIgnorePattern: ^_, caughtErrorsIgnorePattern: ^_, destructuredArrayIgnorePattern: ^_}]
|
||||||
|
"@typescript-eslint/no-use-before-define": [0]
|
||||||
|
"@typescript-eslint/no-useless-constructor": [0]
|
||||||
|
"@typescript-eslint/no-useless-empty-export": [0]
|
||||||
|
"@typescript-eslint/no-wrapper-object-types": [2]
|
||||||
|
"@typescript-eslint/non-nullable-type-assertion-style": [0]
|
||||||
|
"@typescript-eslint/only-throw-error": [2]
|
||||||
|
"@typescript-eslint/parameter-properties": [0]
|
||||||
|
"@typescript-eslint/prefer-as-const": [2]
|
||||||
|
"@typescript-eslint/prefer-destructuring": [0]
|
||||||
|
"@typescript-eslint/prefer-enum-initializers": [0]
|
||||||
|
"@typescript-eslint/prefer-find": [2]
|
||||||
|
"@typescript-eslint/prefer-for-of": [2]
|
||||||
|
"@typescript-eslint/prefer-function-type": [2]
|
||||||
|
"@typescript-eslint/prefer-includes": [2]
|
||||||
|
"@typescript-eslint/prefer-literal-enum-member": [0]
|
||||||
|
"@typescript-eslint/prefer-namespace-keyword": [0]
|
||||||
|
"@typescript-eslint/prefer-nullish-coalescing": [0]
|
||||||
|
"@typescript-eslint/prefer-optional-chain": [2, {requireNullish: true}]
|
||||||
|
"@typescript-eslint/prefer-promise-reject-errors": [0]
|
||||||
|
"@typescript-eslint/prefer-readonly": [0]
|
||||||
|
"@typescript-eslint/prefer-readonly-parameter-types": [0]
|
||||||
|
"@typescript-eslint/prefer-reduce-type-parameter": [0]
|
||||||
|
"@typescript-eslint/prefer-regexp-exec": [0]
|
||||||
|
"@typescript-eslint/prefer-return-this-type": [0]
|
||||||
|
"@typescript-eslint/prefer-string-starts-ends-with": [2, {allowSingleElementEquality: always}]
|
||||||
|
"@typescript-eslint/promise-function-async": [0]
|
||||||
|
"@typescript-eslint/require-array-sort-compare": [0]
|
||||||
|
"@typescript-eslint/require-await": [0]
|
||||||
|
"@typescript-eslint/restrict-plus-operands": [2]
|
||||||
|
"@typescript-eslint/restrict-template-expressions": [0]
|
||||||
|
"@typescript-eslint/return-await": [0]
|
||||||
|
"@typescript-eslint/strict-boolean-expressions": [0]
|
||||||
|
"@typescript-eslint/switch-exhaustiveness-check": [0]
|
||||||
|
"@typescript-eslint/triple-slash-reference": [2]
|
||||||
|
"@typescript-eslint/typedef": [0]
|
||||||
|
"@typescript-eslint/unbound-method": [0] # too many false-positives
|
||||||
|
"@typescript-eslint/unified-signatures": [2]
|
||||||
accessor-pairs: [2]
|
accessor-pairs: [2]
|
||||||
array-callback-return: [2, {checkForEach: true}]
|
array-callback-return: [2, {checkForEach: true}]
|
||||||
array-func/avoid-reverse: [2]
|
array-func/avoid-reverse: [2]
|
||||||
@ -199,6 +359,7 @@ rules:
|
|||||||
default-case-last: [2]
|
default-case-last: [2]
|
||||||
default-case: [0]
|
default-case: [0]
|
||||||
default-param-last: [0]
|
default-param-last: [0]
|
||||||
|
deprecation/deprecation: [2]
|
||||||
dot-notation: [0]
|
dot-notation: [0]
|
||||||
eqeqeq: [2]
|
eqeqeq: [2]
|
||||||
for-direction: [2]
|
for-direction: [2]
|
||||||
@ -260,7 +421,7 @@ rules:
|
|||||||
i/no-internal-modules: [0]
|
i/no-internal-modules: [0]
|
||||||
i/no-mutable-exports: [0]
|
i/no-mutable-exports: [0]
|
||||||
i/no-named-as-default-member: [0]
|
i/no-named-as-default-member: [0]
|
||||||
i/no-named-as-default: [2]
|
i/no-named-as-default: [0]
|
||||||
i/no-named-default: [0]
|
i/no-named-default: [0]
|
||||||
i/no-named-export: [0]
|
i/no-named-export: [0]
|
||||||
i/no-namespace: [0]
|
i/no-namespace: [0]
|
||||||
@ -270,7 +431,7 @@ rules:
|
|||||||
i/no-restricted-paths: [0]
|
i/no-restricted-paths: [0]
|
||||||
i/no-self-import: [2]
|
i/no-self-import: [2]
|
||||||
i/no-unassigned-import: [0]
|
i/no-unassigned-import: [0]
|
||||||
i/no-unresolved: [2, {commonjs: true, ignore: ["\\?.+$", ^vitest/]}]
|
i/no-unresolved: [2, {commonjs: true, ignore: ["\\?.+$"]}]
|
||||||
i/no-unused-modules: [2, {unusedExports: true}]
|
i/no-unused-modules: [2, {unusedExports: true}]
|
||||||
i/no-useless-path-segments: [2, {commonjs: true}]
|
i/no-useless-path-segments: [2, {commonjs: true}]
|
||||||
i/no-webpack-loader-syntax: [2]
|
i/no-webpack-loader-syntax: [2]
|
||||||
@ -278,55 +439,6 @@ rules:
|
|||||||
i/prefer-default-export: [0]
|
i/prefer-default-export: [0]
|
||||||
i/unambiguous: [0]
|
i/unambiguous: [0]
|
||||||
init-declarations: [0]
|
init-declarations: [0]
|
||||||
jquery/no-ajax-events: [2]
|
|
||||||
jquery/no-ajax: [2]
|
|
||||||
jquery/no-animate: [2]
|
|
||||||
jquery/no-attr: [0]
|
|
||||||
jquery/no-bind: [2]
|
|
||||||
jquery/no-class: [0]
|
|
||||||
jquery/no-clone: [2]
|
|
||||||
jquery/no-closest: [0]
|
|
||||||
jquery/no-css: [2]
|
|
||||||
jquery/no-data: [0]
|
|
||||||
jquery/no-deferred: [2]
|
|
||||||
jquery/no-delegate: [2]
|
|
||||||
jquery/no-each: [0]
|
|
||||||
jquery/no-extend: [2]
|
|
||||||
jquery/no-fade: [2]
|
|
||||||
jquery/no-filter: [0]
|
|
||||||
jquery/no-find: [0]
|
|
||||||
jquery/no-global-eval: [2]
|
|
||||||
jquery/no-grep: [2]
|
|
||||||
jquery/no-has: [2]
|
|
||||||
jquery/no-hide: [2]
|
|
||||||
jquery/no-html: [0]
|
|
||||||
jquery/no-in-array: [2]
|
|
||||||
jquery/no-is-array: [2]
|
|
||||||
jquery/no-is-function: [2]
|
|
||||||
jquery/no-is: [2]
|
|
||||||
jquery/no-load: [2]
|
|
||||||
jquery/no-map: [2]
|
|
||||||
jquery/no-merge: [2]
|
|
||||||
jquery/no-param: [2]
|
|
||||||
jquery/no-parent: [0]
|
|
||||||
jquery/no-parents: [0]
|
|
||||||
jquery/no-parse-html: [2]
|
|
||||||
jquery/no-prop: [2]
|
|
||||||
jquery/no-proxy: [2]
|
|
||||||
jquery/no-ready: [2]
|
|
||||||
jquery/no-serialize: [2]
|
|
||||||
jquery/no-show: [2]
|
|
||||||
jquery/no-size: [2]
|
|
||||||
jquery/no-sizzle: [0]
|
|
||||||
jquery/no-slide: [0]
|
|
||||||
jquery/no-submit: [0]
|
|
||||||
jquery/no-text: [0]
|
|
||||||
jquery/no-toggle: [2]
|
|
||||||
jquery/no-trigger: [0]
|
|
||||||
jquery/no-trim: [2]
|
|
||||||
jquery/no-val: [0]
|
|
||||||
jquery/no-when: [2]
|
|
||||||
jquery/no-wrap: [2]
|
|
||||||
line-comment-position: [0]
|
line-comment-position: [0]
|
||||||
logical-assignment-operators: [0]
|
logical-assignment-operators: [0]
|
||||||
max-classes-per-file: [0]
|
max-classes-per-file: [0]
|
||||||
@ -339,7 +451,7 @@ rules:
|
|||||||
multiline-comment-style: [2, separate-lines]
|
multiline-comment-style: [2, separate-lines]
|
||||||
new-cap: [0]
|
new-cap: [0]
|
||||||
no-alert: [0]
|
no-alert: [0]
|
||||||
no-array-constructor: [2]
|
no-array-constructor: [0] # handled by @typescript-eslint/no-array-constructor
|
||||||
no-async-promise-executor: [0]
|
no-async-promise-executor: [0]
|
||||||
no-await-in-loop: [0]
|
no-await-in-loop: [0]
|
||||||
no-bitwise: [0]
|
no-bitwise: [0]
|
||||||
@ -364,7 +476,7 @@ rules:
|
|||||||
no-dupe-else-if: [2]
|
no-dupe-else-if: [2]
|
||||||
no-dupe-keys: [2]
|
no-dupe-keys: [2]
|
||||||
no-duplicate-case: [2]
|
no-duplicate-case: [2]
|
||||||
no-duplicate-imports: [2]
|
no-duplicate-imports: [0]
|
||||||
no-else-return: [2]
|
no-else-return: [2]
|
||||||
no-empty-character-class: [2]
|
no-empty-character-class: [2]
|
||||||
no-empty-function: [0]
|
no-empty-function: [0]
|
||||||
@ -383,7 +495,7 @@ rules:
|
|||||||
no-global-assign: [2]
|
no-global-assign: [2]
|
||||||
no-implicit-coercion: [2]
|
no-implicit-coercion: [2]
|
||||||
no-implicit-globals: [0]
|
no-implicit-globals: [0]
|
||||||
no-implied-eval: [2]
|
no-implied-eval: [0] # handled by @typescript-eslint/no-implied-eval
|
||||||
no-import-assign: [2]
|
no-import-assign: [2]
|
||||||
no-inline-comments: [0]
|
no-inline-comments: [0]
|
||||||
no-inner-declarations: [2]
|
no-inner-declarations: [2]
|
||||||
@ -397,12 +509,12 @@ rules:
|
|||||||
no-jquery/no-animate-toggle: [2]
|
no-jquery/no-animate-toggle: [2]
|
||||||
no-jquery/no-animate: [2]
|
no-jquery/no-animate: [2]
|
||||||
no-jquery/no-append-html: [2]
|
no-jquery/no-append-html: [2]
|
||||||
no-jquery/no-attr: [0]
|
no-jquery/no-attr: [2]
|
||||||
no-jquery/no-bind: [2]
|
no-jquery/no-bind: [2]
|
||||||
no-jquery/no-box-model: [2]
|
no-jquery/no-box-model: [2]
|
||||||
no-jquery/no-browser: [2]
|
no-jquery/no-browser: [2]
|
||||||
no-jquery/no-camel-case: [2]
|
no-jquery/no-camel-case: [2]
|
||||||
no-jquery/no-class-state: [0]
|
no-jquery/no-class-state: [2]
|
||||||
no-jquery/no-class: [0]
|
no-jquery/no-class: [0]
|
||||||
no-jquery/no-clone: [2]
|
no-jquery/no-clone: [2]
|
||||||
no-jquery/no-closest: [0]
|
no-jquery/no-closest: [0]
|
||||||
@ -457,8 +569,8 @@ rules:
|
|||||||
no-jquery/no-other-utils: [2]
|
no-jquery/no-other-utils: [2]
|
||||||
no-jquery/no-param: [2]
|
no-jquery/no-param: [2]
|
||||||
no-jquery/no-parent: [0]
|
no-jquery/no-parent: [0]
|
||||||
no-jquery/no-parents: [0]
|
no-jquery/no-parents: [2]
|
||||||
no-jquery/no-parse-html-literal: [0]
|
no-jquery/no-parse-html-literal: [2]
|
||||||
no-jquery/no-parse-html: [2]
|
no-jquery/no-parse-html: [2]
|
||||||
no-jquery/no-parse-json: [2]
|
no-jquery/no-parse-json: [2]
|
||||||
no-jquery/no-parse-xml: [2]
|
no-jquery/no-parse-xml: [2]
|
||||||
@ -469,11 +581,11 @@ rules:
|
|||||||
no-jquery/no-selector-prop: [2]
|
no-jquery/no-selector-prop: [2]
|
||||||
no-jquery/no-serialize: [2]
|
no-jquery/no-serialize: [2]
|
||||||
no-jquery/no-size: [2]
|
no-jquery/no-size: [2]
|
||||||
no-jquery/no-sizzle: [0]
|
no-jquery/no-sizzle: [2]
|
||||||
no-jquery/no-slide: [2]
|
no-jquery/no-slide: [2]
|
||||||
no-jquery/no-sub: [2]
|
no-jquery/no-sub: [2]
|
||||||
no-jquery/no-support: [2]
|
no-jquery/no-support: [2]
|
||||||
no-jquery/no-text: [0]
|
no-jquery/no-text: [2]
|
||||||
no-jquery/no-trigger: [0]
|
no-jquery/no-trigger: [0]
|
||||||
no-jquery/no-trim: [2]
|
no-jquery/no-trim: [2]
|
||||||
no-jquery/no-type: [2]
|
no-jquery/no-type: [2]
|
||||||
@ -511,12 +623,12 @@ rules:
|
|||||||
no-promise-executor-return: [0]
|
no-promise-executor-return: [0]
|
||||||
no-proto: [2]
|
no-proto: [2]
|
||||||
no-prototype-builtins: [2]
|
no-prototype-builtins: [2]
|
||||||
no-redeclare: [2]
|
no-redeclare: [0] # must be disabled for typescript overloads
|
||||||
no-regex-spaces: [2]
|
no-regex-spaces: [2]
|
||||||
no-restricted-exports: [0]
|
no-restricted-exports: [0]
|
||||||
no-restricted-globals: [2, addEventListener, blur, close, closed, confirm, defaultStatus, defaultstatus, error, event, external, find, focus, frameElement, frames, history, innerHeight, innerWidth, isFinite, isNaN, length, location, locationbar, menubar, moveBy, moveTo, name, onblur, onerror, onfocus, onload, onresize, onunload, open, opener, opera, outerHeight, outerWidth, pageXOffset, pageYOffset, parent, print, removeEventListener, resizeBy, resizeTo, screen, screenLeft, screenTop, screenX, screenY, scroll, scrollbars, scrollBy, scrollTo, scrollX, scrollY, self, status, statusbar, stop, toolbar, top, __dirname, __filename]
|
no-restricted-globals: [2, addEventListener, blur, close, closed, confirm, defaultStatus, defaultstatus, error, event, external, find, focus, frameElement, frames, history, innerHeight, innerWidth, isFinite, isNaN, length, location, locationbar, menubar, moveBy, moveTo, name, onblur, onerror, onfocus, onload, onresize, onunload, open, opener, opera, outerHeight, outerWidth, pageXOffset, pageYOffset, parent, print, removeEventListener, resizeBy, resizeTo, screen, screenLeft, screenTop, screenX, screenY, scroll, scrollbars, scrollBy, scrollTo, scrollX, scrollY, self, status, statusbar, stop, toolbar, top, __dirname, __filename]
|
||||||
no-restricted-imports: [0]
|
no-restricted-imports: [0]
|
||||||
no-restricted-syntax: [2, WithStatement, ForInStatement, LabeledStatement, SequenceExpression, {selector: "CallExpression[callee.name='fetch']", message: "use modules/fetch.js instead"}]
|
no-restricted-syntax: [2, WithStatement, ForInStatement, LabeledStatement, SequenceExpression, {selector: "CallExpression[callee.name='fetch']", message: "use modules/fetch.ts instead"}]
|
||||||
no-return-assign: [0]
|
no-return-assign: [0]
|
||||||
no-script-url: [2]
|
no-script-url: [2]
|
||||||
no-self-assign: [2, {props: true}]
|
no-self-assign: [2, {props: true}]
|
||||||
@ -536,7 +648,7 @@ rules:
|
|||||||
no-underscore-dangle: [0]
|
no-underscore-dangle: [0]
|
||||||
no-unexpected-multiline: [2]
|
no-unexpected-multiline: [2]
|
||||||
no-unmodified-loop-condition: [2]
|
no-unmodified-loop-condition: [2]
|
||||||
no-unneeded-ternary: [0]
|
no-unneeded-ternary: [2]
|
||||||
no-unreachable-loop: [2]
|
no-unreachable-loop: [2]
|
||||||
no-unreachable: [2]
|
no-unreachable: [2]
|
||||||
no-unsafe-finally: [2]
|
no-unsafe-finally: [2]
|
||||||
@ -544,7 +656,7 @@ rules:
|
|||||||
no-unused-expressions: [2]
|
no-unused-expressions: [2]
|
||||||
no-unused-labels: [2]
|
no-unused-labels: [2]
|
||||||
no-unused-private-class-members: [2]
|
no-unused-private-class-members: [2]
|
||||||
no-unused-vars: [2, {args: all, argsIgnorePattern: ^_, varsIgnorePattern: ^_, caughtErrorsIgnorePattern: ^_, destructuredArrayIgnorePattern: ^_, ignoreRestSiblings: false}]
|
no-unused-vars: [0] # handled by @typescript-eslint/no-unused-vars
|
||||||
no-use-before-define: [2, {functions: false, classes: true, variables: true, allowNamedExports: true}]
|
no-use-before-define: [2, {functions: false, classes: true, variables: true, allowNamedExports: true}]
|
||||||
no-use-extend-native/no-use-extend-native: [2]
|
no-use-extend-native/no-use-extend-native: [2]
|
||||||
no-useless-backreference: [2]
|
no-useless-backreference: [2]
|
||||||
@ -659,7 +771,7 @@ rules:
|
|||||||
regexp/unicode-escape: [0]
|
regexp/unicode-escape: [0]
|
||||||
regexp/use-ignore-case: [0]
|
regexp/use-ignore-case: [0]
|
||||||
require-atomic-updates: [0]
|
require-atomic-updates: [0]
|
||||||
require-await: [0]
|
require-await: [0] # handled by @typescript-eslint/require-await
|
||||||
require-unicode-regexp: [0]
|
require-unicode-regexp: [0]
|
||||||
require-yield: [2]
|
require-yield: [2]
|
||||||
sonarjs/cognitive-complexity: [0]
|
sonarjs/cognitive-complexity: [0]
|
||||||
@ -703,6 +815,7 @@ rules:
|
|||||||
unicorn/better-regex: [0]
|
unicorn/better-regex: [0]
|
||||||
unicorn/catch-error-name: [0]
|
unicorn/catch-error-name: [0]
|
||||||
unicorn/consistent-destructuring: [2]
|
unicorn/consistent-destructuring: [2]
|
||||||
|
unicorn/consistent-empty-array-spread: [2]
|
||||||
unicorn/consistent-function-scoping: [2]
|
unicorn/consistent-function-scoping: [2]
|
||||||
unicorn/custom-error-definition: [0]
|
unicorn/custom-error-definition: [0]
|
||||||
unicorn/empty-brace-spaces: [2]
|
unicorn/empty-brace-spaces: [2]
|
||||||
@ -715,28 +828,35 @@ rules:
|
|||||||
unicorn/import-style: [0]
|
unicorn/import-style: [0]
|
||||||
unicorn/new-for-builtins: [2]
|
unicorn/new-for-builtins: [2]
|
||||||
unicorn/no-abusive-eslint-disable: [0]
|
unicorn/no-abusive-eslint-disable: [0]
|
||||||
|
unicorn/no-anonymous-default-export: [0]
|
||||||
unicorn/no-array-callback-reference: [0]
|
unicorn/no-array-callback-reference: [0]
|
||||||
unicorn/no-array-for-each: [2]
|
unicorn/no-array-for-each: [2]
|
||||||
unicorn/no-array-method-this-argument: [2]
|
unicorn/no-array-method-this-argument: [2]
|
||||||
unicorn/no-array-push-push: [2]
|
unicorn/no-array-push-push: [2]
|
||||||
unicorn/no-array-reduce: [2]
|
unicorn/no-array-reduce: [2]
|
||||||
unicorn/no-await-expression-member: [0]
|
unicorn/no-await-expression-member: [0]
|
||||||
|
unicorn/no-await-in-promise-methods: [2]
|
||||||
unicorn/no-console-spaces: [0]
|
unicorn/no-console-spaces: [0]
|
||||||
unicorn/no-document-cookie: [2]
|
unicorn/no-document-cookie: [2]
|
||||||
unicorn/no-empty-file: [2]
|
unicorn/no-empty-file: [2]
|
||||||
unicorn/no-for-loop: [0]
|
unicorn/no-for-loop: [0]
|
||||||
unicorn/no-hex-escape: [0]
|
unicorn/no-hex-escape: [0]
|
||||||
unicorn/no-instanceof-array: [0]
|
unicorn/no-instanceof-array: [0]
|
||||||
|
unicorn/no-invalid-fetch-options: [2]
|
||||||
unicorn/no-invalid-remove-event-listener: [2]
|
unicorn/no-invalid-remove-event-listener: [2]
|
||||||
unicorn/no-keyword-prefix: [0]
|
unicorn/no-keyword-prefix: [0]
|
||||||
|
unicorn/no-length-as-slice-end: [2]
|
||||||
unicorn/no-lonely-if: [2]
|
unicorn/no-lonely-if: [2]
|
||||||
|
unicorn/no-magic-array-flat-depth: [0]
|
||||||
unicorn/no-negated-condition: [0]
|
unicorn/no-negated-condition: [0]
|
||||||
|
unicorn/no-negation-in-equality-check: [2]
|
||||||
unicorn/no-nested-ternary: [0]
|
unicorn/no-nested-ternary: [0]
|
||||||
unicorn/no-new-array: [0]
|
unicorn/no-new-array: [0]
|
||||||
unicorn/no-new-buffer: [0]
|
unicorn/no-new-buffer: [0]
|
||||||
unicorn/no-null: [0]
|
unicorn/no-null: [0]
|
||||||
unicorn/no-object-as-default-parameter: [0]
|
unicorn/no-object-as-default-parameter: [0]
|
||||||
unicorn/no-process-exit: [0]
|
unicorn/no-process-exit: [0]
|
||||||
|
unicorn/no-single-promise-in-promise-methods: [2]
|
||||||
unicorn/no-static-only-class: [2]
|
unicorn/no-static-only-class: [2]
|
||||||
unicorn/no-thenable: [2]
|
unicorn/no-thenable: [2]
|
||||||
unicorn/no-this-assignment: [2]
|
unicorn/no-this-assignment: [2]
|
||||||
@ -788,16 +908,18 @@ rules:
|
|||||||
unicorn/prefer-object-has-own: [0]
|
unicorn/prefer-object-has-own: [0]
|
||||||
unicorn/prefer-optional-catch-binding: [2]
|
unicorn/prefer-optional-catch-binding: [2]
|
||||||
unicorn/prefer-prototype-methods: [0]
|
unicorn/prefer-prototype-methods: [0]
|
||||||
unicorn/prefer-query-selector: [0]
|
unicorn/prefer-query-selector: [2]
|
||||||
unicorn/prefer-reflect-apply: [0]
|
unicorn/prefer-reflect-apply: [0]
|
||||||
unicorn/prefer-regexp-test: [2]
|
unicorn/prefer-regexp-test: [2]
|
||||||
unicorn/prefer-set-has: [0]
|
unicorn/prefer-set-has: [0]
|
||||||
unicorn/prefer-set-size: [2]
|
unicorn/prefer-set-size: [2]
|
||||||
unicorn/prefer-spread: [0]
|
unicorn/prefer-spread: [0]
|
||||||
|
unicorn/prefer-string-raw: [0]
|
||||||
unicorn/prefer-string-replace-all: [0]
|
unicorn/prefer-string-replace-all: [0]
|
||||||
unicorn/prefer-string-slice: [0]
|
unicorn/prefer-string-slice: [0]
|
||||||
unicorn/prefer-string-starts-ends-with: [2]
|
unicorn/prefer-string-starts-ends-with: [2]
|
||||||
unicorn/prefer-string-trim-start-end: [2]
|
unicorn/prefer-string-trim-start-end: [2]
|
||||||
|
unicorn/prefer-structured-clone: [2]
|
||||||
unicorn/prefer-switch: [0]
|
unicorn/prefer-switch: [0]
|
||||||
unicorn/prefer-ternary: [0]
|
unicorn/prefer-ternary: [0]
|
||||||
unicorn/prefer-text-content: [2]
|
unicorn/prefer-text-content: [2]
|
||||||
|
1
.gitattributes
vendored
1
.gitattributes
vendored
@ -1,5 +1,6 @@
|
|||||||
* text=auto eol=lf
|
* text=auto eol=lf
|
||||||
*.tmpl linguist-language=Handlebars
|
*.tmpl linguist-language=Handlebars
|
||||||
|
*.pb.go linguist-generated
|
||||||
/assets/*.json linguist-generated
|
/assets/*.json linguist-generated
|
||||||
/public/assets/img/svg/*.svg linguist-generated
|
/public/assets/img/svg/*.svg linguist-generated
|
||||||
/templates/swagger/v1_json.tmpl linguist-generated
|
/templates/swagger/v1_json.tmpl linguist-generated
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
<!--
|
<!--
|
||||||
1. Please speak English, this is the language all maintainers can speak and write.
|
1. Please speak English, this is the language all maintainers can speak and write.
|
||||||
2. Please ask questions or configuration/deploy problems on our Discord
|
2. Please ask questions or configuration/deploy problems on our Discord
|
||||||
server (https://discord.gg/gitea) or forum (https://discourse.gitea.io).
|
server (https://discord.gg/gitea) or forum (https://forum.gitea.com).
|
||||||
3. Please take a moment to check that your issue doesn't already exist.
|
3. Please take a moment to check that your issue doesn't already exist.
|
||||||
4. Make sure it's not mentioned in the FAQ (https://docs.gitea.com/help/faq)
|
4. Make sure it's not mentioned in the FAQ (https://docs.gitea.com/help/faq)
|
||||||
5. Please give all relevant information below for bug reports, because
|
5. Please give all relevant information below for bug reports, because
|
||||||
@ -21,7 +21,7 @@
|
|||||||
- [ ] MySQL
|
- [ ] MySQL
|
||||||
- [ ] MSSQL
|
- [ ] MSSQL
|
||||||
- [ ] SQLite
|
- [ ] SQLite
|
||||||
- Can you reproduce the bug at https://try.gitea.io:
|
- Can you reproduce the bug at https://demo.gitea.com:
|
||||||
- [ ] Yes (provide example URL)
|
- [ ] Yes (provide example URL)
|
||||||
- [ ] No
|
- [ ] No
|
||||||
- Log gist:
|
- Log gist:
|
||||||
|
6
.github/ISSUE_TEMPLATE/bug-report.yaml
vendored
6
.github/ISSUE_TEMPLATE/bug-report.yaml
vendored
@ -11,7 +11,7 @@ body:
|
|||||||
value: |
|
value: |
|
||||||
1. Please speak English, this is the language all maintainers can speak and write.
|
1. Please speak English, this is the language all maintainers can speak and write.
|
||||||
2. Please ask questions or configuration/deploy problems on our Discord
|
2. Please ask questions or configuration/deploy problems on our Discord
|
||||||
server (https://discord.gg/gitea) or forum (https://discourse.gitea.io).
|
server (https://discord.gg/gitea) or forum (https://forum.gitea.com).
|
||||||
3. Make sure you are using the latest release and
|
3. Make sure you are using the latest release and
|
||||||
take a moment to check that your issue hasn't been reported before.
|
take a moment to check that your issue hasn't been reported before.
|
||||||
4. Make sure it's not mentioned in the FAQ (https://docs.gitea.com/help/faq)
|
4. Make sure it's not mentioned in the FAQ (https://docs.gitea.com/help/faq)
|
||||||
@ -37,7 +37,7 @@ body:
|
|||||||
label: Can you reproduce the bug on the Gitea demo site?
|
label: Can you reproduce the bug on the Gitea demo site?
|
||||||
description: |
|
description: |
|
||||||
If so, please provide a URL in the Description field
|
If so, please provide a URL in the Description field
|
||||||
URL of Gitea demo: https://try.gitea.io
|
URL of Gitea demo: https://demo.gitea.com
|
||||||
options:
|
options:
|
||||||
- "Yes"
|
- "Yes"
|
||||||
- "No"
|
- "No"
|
||||||
@ -74,7 +74,7 @@ body:
|
|||||||
attributes:
|
attributes:
|
||||||
label: How are you running Gitea?
|
label: How are you running Gitea?
|
||||||
description: |
|
description: |
|
||||||
Please include information on whether you built Gitea yourself, used one of our downloads, are using https://try.gitea.io or are using some other package
|
Please include information on whether you built Gitea yourself, used one of our downloads, are using https://demo.gitea.com or are using some other package
|
||||||
Please also tell us how you are running Gitea, e.g. if it is being run from docker, a command-line, systemd etc.
|
Please also tell us how you are running Gitea, e.g. if it is being run from docker, a command-line, systemd etc.
|
||||||
If you are using a package or systemd tell us what distribution you are using
|
If you are using a package or systemd tell us what distribution you are using
|
||||||
validations:
|
validations:
|
||||||
|
2
.github/ISSUE_TEMPLATE/config.yml
vendored
2
.github/ISSUE_TEMPLATE/config.yml
vendored
@ -7,7 +7,7 @@ contact_links:
|
|||||||
url: https://discord.gg/Gitea
|
url: https://discord.gg/Gitea
|
||||||
about: Please ask questions and discuss configuration or deployment problems here.
|
about: Please ask questions and discuss configuration or deployment problems here.
|
||||||
- name: Discourse Forum
|
- name: Discourse Forum
|
||||||
url: https://discourse.gitea.io
|
url: https://forum.gitea.com
|
||||||
about: Questions and configuration or deployment problems can also be discussed on our forum.
|
about: Questions and configuration or deployment problems can also be discussed on our forum.
|
||||||
- name: Frequently Asked Questions
|
- name: Frequently Asked Questions
|
||||||
url: https://docs.gitea.com/help/faq
|
url: https://docs.gitea.com/help/faq
|
||||||
|
2
.github/ISSUE_TEMPLATE/feature-request.yaml
vendored
2
.github/ISSUE_TEMPLATE/feature-request.yaml
vendored
@ -7,7 +7,7 @@ body:
|
|||||||
value: |
|
value: |
|
||||||
1. Please speak English, this is the language all maintainers can speak and write.
|
1. Please speak English, this is the language all maintainers can speak and write.
|
||||||
2. Please ask questions or configuration/deploy problems on our Discord
|
2. Please ask questions or configuration/deploy problems on our Discord
|
||||||
server (https://discord.gg/gitea) or forum (https://discourse.gitea.io).
|
server (https://discord.gg/gitea) or forum (https://forum.gitea.com).
|
||||||
3. Please take a moment to check that your feature hasn't already been suggested.
|
3. Please take a moment to check that your feature hasn't already been suggested.
|
||||||
- type: textarea
|
- type: textarea
|
||||||
id: description
|
id: description
|
||||||
|
4
.github/ISSUE_TEMPLATE/ui.bug-report.yaml
vendored
4
.github/ISSUE_TEMPLATE/ui.bug-report.yaml
vendored
@ -11,7 +11,7 @@ body:
|
|||||||
value: |
|
value: |
|
||||||
1. Please speak English, this is the language all maintainers can speak and write.
|
1. Please speak English, this is the language all maintainers can speak and write.
|
||||||
2. Please ask questions or configuration/deploy problems on our Discord
|
2. Please ask questions or configuration/deploy problems on our Discord
|
||||||
server (https://discord.gg/gitea) or forum (https://discourse.gitea.io).
|
server (https://discord.gg/gitea) or forum (https://forum.gitea.com).
|
||||||
3. Please take a moment to check that your issue doesn't already exist.
|
3. Please take a moment to check that your issue doesn't already exist.
|
||||||
4. Make sure it's not mentioned in the FAQ (https://docs.gitea.com/help/faq)
|
4. Make sure it's not mentioned in the FAQ (https://docs.gitea.com/help/faq)
|
||||||
5. Please give all relevant information below for bug reports, because
|
5. Please give all relevant information below for bug reports, because
|
||||||
@ -46,7 +46,7 @@ body:
|
|||||||
label: Can you reproduce the bug on the Gitea demo site?
|
label: Can you reproduce the bug on the Gitea demo site?
|
||||||
description: |
|
description: |
|
||||||
If so, please provide a URL in the Description field
|
If so, please provide a URL in the Description field
|
||||||
URL of Gitea demo: https://try.gitea.io
|
URL of Gitea demo: https://demo.gitea.com
|
||||||
options:
|
options:
|
||||||
- "Yes"
|
- "Yes"
|
||||||
- "No"
|
- "No"
|
||||||
|
14
.github/labeler.yml
vendored
14
.github/labeler.yml
vendored
@ -4,13 +4,6 @@ modifies/docs:
|
|||||||
- "**/*.md"
|
- "**/*.md"
|
||||||
- "docs/**"
|
- "docs/**"
|
||||||
|
|
||||||
modifies/frontend:
|
|
||||||
- changed-files:
|
|
||||||
- any-glob-to-any-file:
|
|
||||||
- "web_src/**"
|
|
||||||
- "tailwind.config.js"
|
|
||||||
- "webpack.config.js"
|
|
||||||
|
|
||||||
modifies/templates:
|
modifies/templates:
|
||||||
- changed-files:
|
- changed-files:
|
||||||
- all-globs-to-any-file:
|
- all-globs-to-any-file:
|
||||||
@ -53,7 +46,7 @@ modifies/internal:
|
|||||||
- ".gitpod.yml"
|
- ".gitpod.yml"
|
||||||
- ".markdownlint.yaml"
|
- ".markdownlint.yaml"
|
||||||
- ".spectral.yaml"
|
- ".spectral.yaml"
|
||||||
- ".stylelintrc.yaml"
|
- "stylelint.config.js"
|
||||||
- ".yamllint.yaml"
|
- ".yamllint.yaml"
|
||||||
- ".github/**"
|
- ".github/**"
|
||||||
- ".gitea/"
|
- ".gitea/"
|
||||||
@ -82,3 +75,8 @@ modifies/js:
|
|||||||
- any-glob-to-any-file:
|
- any-glob-to-any-file:
|
||||||
- "**/*.js"
|
- "**/*.js"
|
||||||
- "**/*.vue"
|
- "**/*.vue"
|
||||||
|
|
||||||
|
docs-update-needed:
|
||||||
|
- changed-files:
|
||||||
|
- any-glob-to-any-file:
|
||||||
|
- "custom/conf/app.example.ini"
|
||||||
|
11
.github/pull_request_template.md
vendored
11
.github/pull_request_template.md
vendored
@ -1,9 +1,10 @@
|
|||||||
<!-- start tips -->
|
<!-- start tips -->
|
||||||
Please check the following:
|
Please check the following:
|
||||||
1. Make sure you are targeting the `main` branch, pull requests on release branches are only allowed for backports.
|
1. Make sure you are targeting the `main` branch, pull requests on release branches are only allowed for backports.
|
||||||
2. Make sure you have read contributing guidelines: https://github.com/go-gitea/gitea/blob/main/CONTRIBUTING.md .
|
2. Make sure you have read contributing guidelines: https://github.com/go-gitea/gitea/blob/main/CONTRIBUTING.md .
|
||||||
3. Describe what your pull request does and which issue you're targeting (if any).
|
3. For documentations contribution, please go to https://gitea.com/gitea/docs
|
||||||
4. It is recommended to enable "Allow edits by maintainers", so maintainers can help more easily.
|
4. Describe what your pull request does and which issue you're targeting (if any).
|
||||||
5. Your input here will be included in the commit message when this PR has been merged. If you don't want some content to be included, please separate them with a line like `---`.
|
5. It is recommended to enable "Allow edits by maintainers", so maintainers can help more easily.
|
||||||
6. Delete all these tips before posting.
|
6. Your input here will be included in the commit message when this PR has been merged. If you don't want some content to be included, please separate them with a line like `---`.
|
||||||
|
7. Delete all these tips before posting.
|
||||||
<!-- end tips -->
|
<!-- end tips -->
|
||||||
|
33
.github/workflows/cron-translations.yml
vendored
33
.github/workflows/cron-translations.yml
vendored
@ -11,14 +11,19 @@ jobs:
|
|||||||
if: github.repository == 'go-gitea/gitea'
|
if: github.repository == 'go-gitea/gitea'
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- name: download from crowdin
|
- uses: crowdin/github-action@v1
|
||||||
uses: docker://jonasfranz/crowdin
|
with:
|
||||||
|
upload_sources: true
|
||||||
|
upload_translations: false
|
||||||
|
download_sources: false
|
||||||
|
download_translations: true
|
||||||
|
push_translations: false
|
||||||
|
push_sources: false
|
||||||
|
create_pull_request: false
|
||||||
|
config: crowdin.yml
|
||||||
env:
|
env:
|
||||||
|
CROWDIN_PROJECT_ID: ${{ secrets.CROWDIN_PROJECT_ID }}
|
||||||
CROWDIN_KEY: ${{ secrets.CROWDIN_KEY }}
|
CROWDIN_KEY: ${{ secrets.CROWDIN_KEY }}
|
||||||
PLUGIN_DOWNLOAD: true
|
|
||||||
PLUGIN_EXPORT_DIR: options/locale/
|
|
||||||
PLUGIN_IGNORE_BRANCH: true
|
|
||||||
PLUGIN_PROJECT_IDENTIFIER: gitea
|
|
||||||
- name: update locales
|
- name: update locales
|
||||||
run: ./build/update-locales.sh
|
run: ./build/update-locales.sh
|
||||||
- name: push translations to repo
|
- name: push translations to repo
|
||||||
@ -31,19 +36,3 @@ jobs:
|
|||||||
commit_message: "[skip ci] Updated translations via Crowdin"
|
commit_message: "[skip ci] Updated translations via Crowdin"
|
||||||
remote: "git@github.com:go-gitea/gitea.git"
|
remote: "git@github.com:go-gitea/gitea.git"
|
||||||
ssh_key: ${{ secrets.DEPLOY_KEY }}
|
ssh_key: ${{ secrets.DEPLOY_KEY }}
|
||||||
crowdin-push:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
if: github.repository == 'go-gitea/gitea'
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
- name: push translations to crowdin
|
|
||||||
uses: docker://jonasfranz/crowdin
|
|
||||||
env:
|
|
||||||
CROWDIN_KEY: ${{ secrets.CROWDIN_KEY }}
|
|
||||||
PLUGIN_UPLOAD: true
|
|
||||||
PLUGIN_EXPORT_DIR: options/locale/
|
|
||||||
PLUGIN_IGNORE_BRANCH: true
|
|
||||||
PLUGIN_PROJECT_IDENTIFIER: gitea
|
|
||||||
PLUGIN_FILES: |
|
|
||||||
locale_en-US.ini: options/locale/locale_en-US.ini
|
|
||||||
PLUGIN_BRANCH: main
|
|
||||||
|
25
.github/workflows/disk-clean.yml
vendored
25
.github/workflows/disk-clean.yml
vendored
@ -1,25 +0,0 @@
|
|||||||
name: disk-clean
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_call:
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
triage:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
- name: Free Disk Space (Ubuntu)
|
|
||||||
uses: jlumbroso/free-disk-space@main
|
|
||||||
with:
|
|
||||||
# this might remove tools that are actually needed,
|
|
||||||
# if set to "true" but frees about 6 GB
|
|
||||||
tool-cache: false
|
|
||||||
|
|
||||||
# all of these default to true, but feel free to set to
|
|
||||||
# "false" if necessary for your workflow
|
|
||||||
android: true
|
|
||||||
dotnet: true
|
|
||||||
haskell: true
|
|
||||||
large-packages: false
|
|
||||||
docker-images: false
|
|
||||||
swap-storage: true
|
|
4
.github/workflows/files-changed.yml
vendored
4
.github/workflows/files-changed.yml
vendored
@ -58,12 +58,11 @@ jobs:
|
|||||||
- "package-lock.json"
|
- "package-lock.json"
|
||||||
- "Makefile"
|
- "Makefile"
|
||||||
- ".eslintrc.yaml"
|
- ".eslintrc.yaml"
|
||||||
- ".stylelintrc.yaml"
|
- "stylelint.config.js"
|
||||||
- ".npmrc"
|
- ".npmrc"
|
||||||
|
|
||||||
docs:
|
docs:
|
||||||
- "**/*.md"
|
- "**/*.md"
|
||||||
- "docs/**"
|
|
||||||
- ".markdownlint.yaml"
|
- ".markdownlint.yaml"
|
||||||
- "package.json"
|
- "package.json"
|
||||||
- "package-lock.json"
|
- "package-lock.json"
|
||||||
@ -73,6 +72,7 @@ jobs:
|
|||||||
- "Makefile"
|
- "Makefile"
|
||||||
|
|
||||||
templates:
|
templates:
|
||||||
|
- "tools/lint-templates-*.js"
|
||||||
- "templates/**/*.tmpl"
|
- "templates/**/*.tmpl"
|
||||||
- "pyproject.toml"
|
- "pyproject.toml"
|
||||||
- "poetry.lock"
|
- "poetry.lock"
|
||||||
|
13
.github/workflows/pull-compliance.yml
vendored
13
.github/workflows/pull-compliance.yml
vendored
@ -35,8 +35,14 @@ jobs:
|
|||||||
- uses: actions/setup-python@v5
|
- uses: actions/setup-python@v5
|
||||||
with:
|
with:
|
||||||
python-version: "3.12"
|
python-version: "3.12"
|
||||||
|
- uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: 20
|
||||||
|
cache: npm
|
||||||
|
cache-dependency-path: package-lock.json
|
||||||
- run: pip install poetry
|
- run: pip install poetry
|
||||||
- run: make deps-py
|
- run: make deps-py
|
||||||
|
- run: make deps-frontend
|
||||||
- run: make lint-templates
|
- run: make lint-templates
|
||||||
|
|
||||||
lint-yaml:
|
lint-yaml:
|
||||||
@ -61,6 +67,8 @@ jobs:
|
|||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 20
|
||||||
|
cache: npm
|
||||||
|
cache-dependency-path: package-lock.json
|
||||||
- run: make deps-frontend
|
- run: make deps-frontend
|
||||||
- run: make lint-swagger
|
- run: make lint-swagger
|
||||||
|
|
||||||
@ -130,6 +138,8 @@ jobs:
|
|||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 20
|
||||||
|
cache: npm
|
||||||
|
cache-dependency-path: package-lock.json
|
||||||
- run: make deps-frontend
|
- run: make deps-frontend
|
||||||
- run: make lint-frontend
|
- run: make lint-frontend
|
||||||
- run: make checks-frontend
|
- run: make checks-frontend
|
||||||
@ -177,9 +187,10 @@ jobs:
|
|||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 20
|
||||||
|
cache: npm
|
||||||
|
cache-dependency-path: package-lock.json
|
||||||
- run: make deps-frontend
|
- run: make deps-frontend
|
||||||
- run: make lint-md
|
- run: make lint-md
|
||||||
- run: make docs
|
|
||||||
|
|
||||||
actions:
|
actions:
|
||||||
if: needs.files-changed.outputs.actions == 'true' || needs.files-changed.outputs.actions == 'true'
|
if: needs.files-changed.outputs.actions == 'true' || needs.files-changed.outputs.actions == 'true'
|
||||||
|
12
.github/workflows/pull-db-tests.yml
vendored
12
.github/workflows/pull-db-tests.yml
vendored
@ -119,6 +119,10 @@ jobs:
|
|||||||
MINIO_SECRET_KEY: 12345678
|
MINIO_SECRET_KEY: 12345678
|
||||||
ports:
|
ports:
|
||||||
- "9000:9000"
|
- "9000:9000"
|
||||||
|
devstoreaccount1.azurite.local: # https://github.com/Azure/Azurite/issues/1583
|
||||||
|
image: mcr.microsoft.com/azure-storage/azurite:latest
|
||||||
|
ports:
|
||||||
|
- 10000:10000
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- uses: actions/setup-go@v5
|
- uses: actions/setup-go@v5
|
||||||
@ -126,7 +130,7 @@ jobs:
|
|||||||
go-version-file: go.mod
|
go-version-file: go.mod
|
||||||
check-latest: true
|
check-latest: true
|
||||||
- name: Add hosts to /etc/hosts
|
- name: Add hosts to /etc/hosts
|
||||||
run: '[ -e "/.dockerenv" ] || [ -e "/run/.containerenv" ] || echo "127.0.0.1 mysql elasticsearch meilisearch smtpimap" | sudo tee -a /etc/hosts'
|
run: '[ -e "/.dockerenv" ] || [ -e "/run/.containerenv" ] || echo "127.0.0.1 minio devstoreaccount1.azurite.local mysql elasticsearch meilisearch smtpimap" | sudo tee -a /etc/hosts'
|
||||||
- run: make deps-backend
|
- run: make deps-backend
|
||||||
- run: make backend
|
- run: make backend
|
||||||
env:
|
env:
|
||||||
@ -204,6 +208,10 @@ jobs:
|
|||||||
SA_PASSWORD: MwantsaSecurePassword1
|
SA_PASSWORD: MwantsaSecurePassword1
|
||||||
ports:
|
ports:
|
||||||
- "1433:1433"
|
- "1433:1433"
|
||||||
|
devstoreaccount1.azurite.local: # https://github.com/Azure/Azurite/issues/1583
|
||||||
|
image: mcr.microsoft.com/azure-storage/azurite:latest
|
||||||
|
ports:
|
||||||
|
- 10000:10000
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- uses: actions/setup-go@v5
|
- uses: actions/setup-go@v5
|
||||||
@ -211,7 +219,7 @@ jobs:
|
|||||||
go-version-file: go.mod
|
go-version-file: go.mod
|
||||||
check-latest: true
|
check-latest: true
|
||||||
- name: Add hosts to /etc/hosts
|
- name: Add hosts to /etc/hosts
|
||||||
run: '[ -e "/.dockerenv" ] || [ -e "/run/.containerenv" ] || echo "127.0.0.1 mssql" | sudo tee -a /etc/hosts'
|
run: '[ -e "/.dockerenv" ] || [ -e "/run/.containerenv" ] || echo "127.0.0.1 mssql devstoreaccount1.azurite.local" | sudo tee -a /etc/hosts'
|
||||||
- run: make deps-backend
|
- run: make deps-backend
|
||||||
- run: make backend
|
- run: make backend
|
||||||
env:
|
env:
|
||||||
|
2
.github/workflows/pull-e2e-tests.yml
vendored
2
.github/workflows/pull-e2e-tests.yml
vendored
@ -24,6 +24,8 @@ jobs:
|
|||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 20
|
||||||
|
cache: npm
|
||||||
|
cache-dependency-path: package-lock.json
|
||||||
- run: make deps-frontend frontend deps-backend
|
- run: make deps-frontend frontend deps-backend
|
||||||
- run: npx playwright install --with-deps
|
- run: npx playwright install --with-deps
|
||||||
- run: make test-e2e-sqlite
|
- run: make test-e2e-sqlite
|
||||||
|
6
.github/workflows/release-nightly.yml
vendored
6
.github/workflows/release-nightly.yml
vendored
@ -9,8 +9,6 @@ concurrency:
|
|||||||
cancel-in-progress: true
|
cancel-in-progress: true
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
disk-clean:
|
|
||||||
uses: ./.github/workflows/disk-clean.yml
|
|
||||||
nightly-binary:
|
nightly-binary:
|
||||||
runs-on: nscloud
|
runs-on: nscloud
|
||||||
steps:
|
steps:
|
||||||
@ -25,6 +23,8 @@ jobs:
|
|||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 20
|
||||||
|
cache: npm
|
||||||
|
cache-dependency-path: package-lock.json
|
||||||
- run: make deps-frontend deps-backend
|
- run: make deps-frontend deps-backend
|
||||||
# xgo build
|
# xgo build
|
||||||
- run: make release
|
- run: make release
|
||||||
@ -47,7 +47,7 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
REF_NAME=$(echo "${{ github.ref }}" | sed -e 's/refs\/heads\///' -e 's/refs\/tags\///' -e 's/release\/v//')
|
REF_NAME=$(echo "${{ github.ref }}" | sed -e 's/refs\/heads\///' -e 's/refs\/tags\///' -e 's/release\/v//')
|
||||||
echo "Cleaned name is ${REF_NAME}"
|
echo "Cleaned name is ${REF_NAME}"
|
||||||
echo "branch=${REF_NAME}" >> "$GITHUB_OUTPUT"
|
echo "branch=${REF_NAME}-nightly" >> "$GITHUB_OUTPUT"
|
||||||
- name: configure aws
|
- name: configure aws
|
||||||
uses: aws-actions/configure-aws-credentials@v4
|
uses: aws-actions/configure-aws-credentials@v4
|
||||||
with:
|
with:
|
||||||
|
2
.github/workflows/release-tag-rc.yml
vendored
2
.github/workflows/release-tag-rc.yml
vendored
@ -24,6 +24,8 @@ jobs:
|
|||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 20
|
||||||
|
cache: npm
|
||||||
|
cache-dependency-path: package-lock.json
|
||||||
- run: make deps-frontend deps-backend
|
- run: make deps-frontend deps-backend
|
||||||
# xgo build
|
# xgo build
|
||||||
- run: make release
|
- run: make release
|
||||||
|
2
.github/workflows/release-tag-version.yml
vendored
2
.github/workflows/release-tag-version.yml
vendored
@ -26,6 +26,8 @@ jobs:
|
|||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 20
|
||||||
|
cache: npm
|
||||||
|
cache-dependency-path: package-lock.json
|
||||||
- run: make deps-frontend deps-backend
|
- run: make deps-frontend deps-backend
|
||||||
# xgo build
|
# xgo build
|
||||||
- run: make release
|
- run: make release
|
||||||
|
7
.gitignore
vendored
7
.gitignore
vendored
@ -77,7 +77,6 @@ cpu.out
|
|||||||
/public/assets/css
|
/public/assets/css
|
||||||
/public/assets/fonts
|
/public/assets/fonts
|
||||||
/public/assets/licenses.txt
|
/public/assets/licenses.txt
|
||||||
/public/assets/img/webpack
|
|
||||||
/vendor
|
/vendor
|
||||||
/web_src/fomantic/node_modules
|
/web_src/fomantic/node_modules
|
||||||
/web_src/fomantic/build/*
|
/web_src/fomantic/build/*
|
||||||
@ -95,6 +94,9 @@ cpu.out
|
|||||||
/.air
|
/.air
|
||||||
/.go-licenses
|
/.go-licenses
|
||||||
|
|
||||||
|
# Files and folders that were previously generated
|
||||||
|
/public/assets/img/webpack
|
||||||
|
|
||||||
# Snapcraft
|
# Snapcraft
|
||||||
/gitea_a*.txt
|
/gitea_a*.txt
|
||||||
snap/.snapcraft/
|
snap/.snapcraft/
|
||||||
@ -106,6 +108,9 @@ prime/
|
|||||||
*_source.tar.bz2
|
*_source.tar.bz2
|
||||||
.DS_Store
|
.DS_Store
|
||||||
|
|
||||||
|
# nix-direnv generated files
|
||||||
|
.direnv/
|
||||||
|
|
||||||
# Make evidence files
|
# Make evidence files
|
||||||
/.make_evidence
|
/.make_evidence
|
||||||
|
|
||||||
|
@ -43,7 +43,7 @@ vscode:
|
|||||||
- Vue.volar
|
- Vue.volar
|
||||||
- ms-azuretools.vscode-docker
|
- ms-azuretools.vscode-docker
|
||||||
- vitest.explorer
|
- vitest.explorer
|
||||||
- qwtel.sqlite-viewer
|
- cweijan.vscode-database-client2
|
||||||
- GitHub.vscode-pull-request-github
|
- GitHub.vscode-pull-request-github
|
||||||
|
|
||||||
ports:
|
ports:
|
||||||
|
@ -1,13 +1,14 @@
|
|||||||
linters:
|
linters:
|
||||||
|
enable-all: false
|
||||||
|
disable-all: true
|
||||||
|
fast: false
|
||||||
enable:
|
enable:
|
||||||
- bidichk
|
- bidichk
|
||||||
# - deadcode # deprecated - https://github.com/golangci/golangci-lint/issues/1841
|
|
||||||
- depguard
|
- depguard
|
||||||
- dupl
|
- dupl
|
||||||
- errcheck
|
- errcheck
|
||||||
- forbidigo
|
- forbidigo
|
||||||
- gocritic
|
- gocritic
|
||||||
# - gocyclo # The cyclomatic complexety of a lot of functions is too high, we should refactor those another time.
|
|
||||||
- gofmt
|
- gofmt
|
||||||
- gofumpt
|
- gofumpt
|
||||||
- gosimple
|
- gosimple
|
||||||
@ -17,23 +18,20 @@ linters:
|
|||||||
- nolintlint
|
- nolintlint
|
||||||
- revive
|
- revive
|
||||||
- staticcheck
|
- staticcheck
|
||||||
# - structcheck # deprecated - https://github.com/golangci/golangci-lint/issues/1841
|
|
||||||
- stylecheck
|
- stylecheck
|
||||||
- typecheck
|
- typecheck
|
||||||
- unconvert
|
- unconvert
|
||||||
- unused
|
- unused
|
||||||
# - varcheck # deprecated - https://github.com/golangci/golangci-lint/issues/1841
|
- unparam
|
||||||
- wastedassign
|
- wastedassign
|
||||||
enable-all: false
|
|
||||||
disable-all: true
|
|
||||||
fast: false
|
|
||||||
|
|
||||||
run:
|
run:
|
||||||
timeout: 10m
|
timeout: 10m
|
||||||
skip-dirs:
|
|
||||||
- node_modules
|
output:
|
||||||
- public
|
sort-results: true
|
||||||
- web_src
|
sort-order: [file]
|
||||||
|
show-stats: true
|
||||||
|
|
||||||
linters-settings:
|
linters-settings:
|
||||||
stylecheck:
|
stylecheck:
|
||||||
@ -45,33 +43,39 @@ linters-settings:
|
|||||||
- ifElseChain
|
- ifElseChain
|
||||||
- singleCaseSwitch # Every time this occurred in the code, there was no other way.
|
- singleCaseSwitch # Every time this occurred in the code, there was no other way.
|
||||||
revive:
|
revive:
|
||||||
ignore-generated-header: false
|
severity: error
|
||||||
severity: warning
|
|
||||||
confidence: 0.8
|
|
||||||
errorCode: 1
|
|
||||||
warningCode: 1
|
|
||||||
rules:
|
rules:
|
||||||
|
- name: atomic
|
||||||
|
- name: bare-return
|
||||||
- name: blank-imports
|
- name: blank-imports
|
||||||
|
- name: constant-logical-expr
|
||||||
- name: context-as-argument
|
- name: context-as-argument
|
||||||
- name: context-keys-type
|
- name: context-keys-type
|
||||||
- name: dot-imports
|
- name: dot-imports
|
||||||
|
- name: duplicated-imports
|
||||||
|
- name: empty-lines
|
||||||
|
- name: error-naming
|
||||||
- name: error-return
|
- name: error-return
|
||||||
- name: error-strings
|
- name: error-strings
|
||||||
- name: error-naming
|
- name: errorf
|
||||||
- name: exported
|
- name: exported
|
||||||
|
- name: identical-branches
|
||||||
- name: if-return
|
- name: if-return
|
||||||
- name: increment-decrement
|
- name: increment-decrement
|
||||||
- name: var-naming
|
- name: indent-error-flow
|
||||||
- name: var-declaration
|
- name: modifies-value-receiver
|
||||||
- name: package-comments
|
- name: package-comments
|
||||||
- name: range
|
- name: range
|
||||||
- name: receiver-naming
|
- name: receiver-naming
|
||||||
|
- name: redefines-builtin-id
|
||||||
|
- name: string-of-int
|
||||||
|
- name: superfluous-else
|
||||||
- name: time-naming
|
- name: time-naming
|
||||||
|
- name: unconditional-recursion
|
||||||
- name: unexported-return
|
- name: unexported-return
|
||||||
- name: indent-error-flow
|
- name: unreachable-code
|
||||||
- name: errorf
|
- name: var-declaration
|
||||||
- name: duplicated-imports
|
- name: var-naming
|
||||||
- name: modifies-value-receiver
|
|
||||||
gofumpt:
|
gofumpt:
|
||||||
extra-rules: true
|
extra-rules: true
|
||||||
depguard:
|
depguard:
|
||||||
@ -90,12 +94,15 @@ linters-settings:
|
|||||||
desc: do not use the internal package, use AddXxx function instead
|
desc: do not use the internal package, use AddXxx function instead
|
||||||
- pkg: gopkg.in/ini.v1
|
- pkg: gopkg.in/ini.v1
|
||||||
desc: do not use the ini package, use gitea's config system instead
|
desc: do not use the ini package, use gitea's config system instead
|
||||||
|
- pkg: gitea.com/go-chi/cache
|
||||||
|
desc: do not use the go-chi cache package, use gitea's cache system
|
||||||
|
|
||||||
issues:
|
issues:
|
||||||
max-issues-per-linter: 0
|
max-issues-per-linter: 0
|
||||||
max-same-issues: 0
|
max-same-issues: 0
|
||||||
|
exclude-dirs: [node_modules, public, web_src]
|
||||||
|
exclude-case-sensitive: true
|
||||||
exclude-rules:
|
exclude-rules:
|
||||||
# Exclude some linters from running on tests files.
|
|
||||||
- path: _test\.go
|
- path: _test\.go
|
||||||
linters:
|
linters:
|
||||||
- gocyclo
|
- gocyclo
|
||||||
@ -113,19 +120,19 @@ issues:
|
|||||||
- path: cmd
|
- path: cmd
|
||||||
linters:
|
linters:
|
||||||
- forbidigo
|
- forbidigo
|
||||||
- linters:
|
- text: "webhook"
|
||||||
|
linters:
|
||||||
- dupl
|
- dupl
|
||||||
text: "webhook"
|
- text: "`ID' should not be capitalized"
|
||||||
- linters:
|
linters:
|
||||||
- gocritic
|
- gocritic
|
||||||
text: "`ID' should not be capitalized"
|
- text: "swagger"
|
||||||
- linters:
|
linters:
|
||||||
- unused
|
- unused
|
||||||
- deadcode
|
- deadcode
|
||||||
text: "swagger"
|
- text: "argument x is overwritten before first use"
|
||||||
- linters:
|
linters:
|
||||||
- staticcheck
|
- staticcheck
|
||||||
text: "argument x is overwritten before first use"
|
|
||||||
- text: "commentFormatting: put a space between `//` and comment text"
|
- text: "commentFormatting: put a space between `//` and comment text"
|
||||||
linters:
|
linters:
|
||||||
- gocritic
|
- gocritic
|
||||||
|
4
.ignore
4
.ignore
@ -4,6 +4,8 @@
|
|||||||
/modules/options/bindata.go
|
/modules/options/bindata.go
|
||||||
/modules/public/bindata.go
|
/modules/public/bindata.go
|
||||||
/modules/templates/bindata.go
|
/modules/templates/bindata.go
|
||||||
/vendor
|
/options/gitignore
|
||||||
|
/options/license
|
||||||
/public/assets
|
/public/assets
|
||||||
|
/vendor
|
||||||
node_modules
|
node_modules
|
||||||
|
@ -1,223 +0,0 @@
|
|||||||
plugins:
|
|
||||||
- stylelint-declaration-strict-value
|
|
||||||
- stylelint-declaration-block-no-ignored-properties
|
|
||||||
- "@stylistic/stylelint-plugin"
|
|
||||||
|
|
||||||
ignoreFiles:
|
|
||||||
- "**/*.go"
|
|
||||||
|
|
||||||
overrides:
|
|
||||||
- files: ["**/chroma/*", "**/codemirror/*", "**/standalone/*", "**/console.css", "font_i18n.css"]
|
|
||||||
rules:
|
|
||||||
scale-unlimited/declaration-strict-value: null
|
|
||||||
- files: ["**/chroma/*", "**/codemirror/*"]
|
|
||||||
rules:
|
|
||||||
block-no-empty: null
|
|
||||||
- files: ["**/*.vue"]
|
|
||||||
customSyntax: postcss-html
|
|
||||||
|
|
||||||
rules:
|
|
||||||
"@stylistic/at-rule-name-case": null
|
|
||||||
"@stylistic/at-rule-name-newline-after": null
|
|
||||||
"@stylistic/at-rule-name-space-after": null
|
|
||||||
"@stylistic/at-rule-semicolon-newline-after": null
|
|
||||||
"@stylistic/at-rule-semicolon-space-before": null
|
|
||||||
"@stylistic/block-closing-brace-empty-line-before": null
|
|
||||||
"@stylistic/block-closing-brace-newline-after": null
|
|
||||||
"@stylistic/block-closing-brace-newline-before": null
|
|
||||||
"@stylistic/block-closing-brace-space-after": null
|
|
||||||
"@stylistic/block-closing-brace-space-before": null
|
|
||||||
"@stylistic/block-opening-brace-newline-after": null
|
|
||||||
"@stylistic/block-opening-brace-newline-before": null
|
|
||||||
"@stylistic/block-opening-brace-space-after": null
|
|
||||||
"@stylistic/block-opening-brace-space-before": always
|
|
||||||
"@stylistic/color-hex-case": lower
|
|
||||||
"@stylistic/declaration-bang-space-after": never
|
|
||||||
"@stylistic/declaration-bang-space-before": null
|
|
||||||
"@stylistic/declaration-block-semicolon-newline-after": null
|
|
||||||
"@stylistic/declaration-block-semicolon-newline-before": null
|
|
||||||
"@stylistic/declaration-block-semicolon-space-after": null
|
|
||||||
"@stylistic/declaration-block-semicolon-space-before": never
|
|
||||||
"@stylistic/declaration-block-trailing-semicolon": null
|
|
||||||
"@stylistic/declaration-colon-newline-after": null
|
|
||||||
"@stylistic/declaration-colon-space-after": null
|
|
||||||
"@stylistic/declaration-colon-space-before": never
|
|
||||||
"@stylistic/function-comma-newline-after": null
|
|
||||||
"@stylistic/function-comma-newline-before": null
|
|
||||||
"@stylistic/function-comma-space-after": null
|
|
||||||
"@stylistic/function-comma-space-before": null
|
|
||||||
"@stylistic/function-max-empty-lines": 0
|
|
||||||
"@stylistic/function-parentheses-newline-inside": never-multi-line
|
|
||||||
"@stylistic/function-parentheses-space-inside": null
|
|
||||||
"@stylistic/function-whitespace-after": null
|
|
||||||
"@stylistic/indentation": 2
|
|
||||||
"@stylistic/linebreaks": null
|
|
||||||
"@stylistic/max-empty-lines": 1
|
|
||||||
"@stylistic/max-line-length": null
|
|
||||||
"@stylistic/media-feature-colon-space-after": null
|
|
||||||
"@stylistic/media-feature-colon-space-before": never
|
|
||||||
"@stylistic/media-feature-name-case": null
|
|
||||||
"@stylistic/media-feature-parentheses-space-inside": null
|
|
||||||
"@stylistic/media-feature-range-operator-space-after": always
|
|
||||||
"@stylistic/media-feature-range-operator-space-before": always
|
|
||||||
"@stylistic/media-query-list-comma-newline-after": null
|
|
||||||
"@stylistic/media-query-list-comma-newline-before": null
|
|
||||||
"@stylistic/media-query-list-comma-space-after": null
|
|
||||||
"@stylistic/media-query-list-comma-space-before": null
|
|
||||||
"@stylistic/named-grid-areas-alignment": null
|
|
||||||
"@stylistic/no-empty-first-line": null
|
|
||||||
"@stylistic/no-eol-whitespace": true
|
|
||||||
"@stylistic/no-extra-semicolons": true
|
|
||||||
"@stylistic/no-missing-end-of-source-newline": null
|
|
||||||
"@stylistic/number-leading-zero": null
|
|
||||||
"@stylistic/number-no-trailing-zeros": null
|
|
||||||
"@stylistic/property-case": lower
|
|
||||||
"@stylistic/selector-attribute-brackets-space-inside": null
|
|
||||||
"@stylistic/selector-attribute-operator-space-after": null
|
|
||||||
"@stylistic/selector-attribute-operator-space-before": null
|
|
||||||
"@stylistic/selector-combinator-space-after": null
|
|
||||||
"@stylistic/selector-combinator-space-before": null
|
|
||||||
"@stylistic/selector-descendant-combinator-no-non-space": null
|
|
||||||
"@stylistic/selector-list-comma-newline-after": null
|
|
||||||
"@stylistic/selector-list-comma-newline-before": null
|
|
||||||
"@stylistic/selector-list-comma-space-after": always-single-line
|
|
||||||
"@stylistic/selector-list-comma-space-before": never-single-line
|
|
||||||
"@stylistic/selector-max-empty-lines": 0
|
|
||||||
"@stylistic/selector-pseudo-class-case": lower
|
|
||||||
"@stylistic/selector-pseudo-class-parentheses-space-inside": never
|
|
||||||
"@stylistic/selector-pseudo-element-case": lower
|
|
||||||
"@stylistic/string-quotes": double
|
|
||||||
"@stylistic/unicode-bom": null
|
|
||||||
"@stylistic/unit-case": lower
|
|
||||||
"@stylistic/value-list-comma-newline-after": null
|
|
||||||
"@stylistic/value-list-comma-newline-before": null
|
|
||||||
"@stylistic/value-list-comma-space-after": null
|
|
||||||
"@stylistic/value-list-comma-space-before": null
|
|
||||||
"@stylistic/value-list-max-empty-lines": 0
|
|
||||||
alpha-value-notation: null
|
|
||||||
annotation-no-unknown: true
|
|
||||||
at-rule-allowed-list: null
|
|
||||||
at-rule-disallowed-list: null
|
|
||||||
at-rule-empty-line-before: null
|
|
||||||
at-rule-no-unknown: [true, {ignoreAtRules: [tailwind]}]
|
|
||||||
at-rule-no-vendor-prefix: true
|
|
||||||
at-rule-property-required-list: null
|
|
||||||
block-no-empty: true
|
|
||||||
color-function-notation: null
|
|
||||||
color-hex-alpha: null
|
|
||||||
color-hex-length: null
|
|
||||||
color-named: null
|
|
||||||
color-no-hex: null
|
|
||||||
color-no-invalid-hex: true
|
|
||||||
comment-empty-line-before: null
|
|
||||||
comment-no-empty: true
|
|
||||||
comment-pattern: null
|
|
||||||
comment-whitespace-inside: null
|
|
||||||
comment-word-disallowed-list: null
|
|
||||||
custom-media-pattern: null
|
|
||||||
custom-property-empty-line-before: null
|
|
||||||
custom-property-no-missing-var-function: true
|
|
||||||
custom-property-pattern: null
|
|
||||||
declaration-block-no-duplicate-custom-properties: true
|
|
||||||
declaration-block-no-duplicate-properties: [true, {ignore: [consecutive-duplicates-with-different-values]}]
|
|
||||||
declaration-block-no-redundant-longhand-properties: null
|
|
||||||
declaration-block-no-shorthand-property-overrides: null
|
|
||||||
declaration-block-single-line-max-declarations: null
|
|
||||||
declaration-empty-line-before: null
|
|
||||||
declaration-no-important: null
|
|
||||||
declaration-property-max-values: null
|
|
||||||
declaration-property-unit-allowed-list: null
|
|
||||||
declaration-property-unit-disallowed-list: {line-height: [em]}
|
|
||||||
declaration-property-value-allowed-list: null
|
|
||||||
declaration-property-value-disallowed-list: null
|
|
||||||
declaration-property-value-no-unknown: true
|
|
||||||
font-family-name-quotes: always-where-recommended
|
|
||||||
font-family-no-duplicate-names: true
|
|
||||||
font-family-no-missing-generic-family-keyword: true
|
|
||||||
font-weight-notation: null
|
|
||||||
function-allowed-list: null
|
|
||||||
function-calc-no-unspaced-operator: true
|
|
||||||
function-disallowed-list: null
|
|
||||||
function-linear-gradient-no-nonstandard-direction: true
|
|
||||||
function-name-case: lower
|
|
||||||
function-no-unknown: true
|
|
||||||
function-url-no-scheme-relative: null
|
|
||||||
function-url-quotes: always
|
|
||||||
function-url-scheme-allowed-list: null
|
|
||||||
function-url-scheme-disallowed-list: null
|
|
||||||
hue-degree-notation: null
|
|
||||||
import-notation: string
|
|
||||||
keyframe-block-no-duplicate-selectors: true
|
|
||||||
keyframe-declaration-no-important: true
|
|
||||||
keyframe-selector-notation: null
|
|
||||||
keyframes-name-pattern: null
|
|
||||||
length-zero-no-unit: [true, ignore: [custom-properties], ignoreFunctions: [var]]
|
|
||||||
max-nesting-depth: null
|
|
||||||
media-feature-name-allowed-list: null
|
|
||||||
media-feature-name-disallowed-list: null
|
|
||||||
media-feature-name-no-unknown: true
|
|
||||||
media-feature-name-no-vendor-prefix: true
|
|
||||||
media-feature-name-unit-allowed-list: null
|
|
||||||
media-feature-name-value-allowed-list: null
|
|
||||||
media-feature-name-value-no-unknown: true
|
|
||||||
media-feature-range-notation: null
|
|
||||||
media-query-no-invalid: true
|
|
||||||
named-grid-areas-no-invalid: true
|
|
||||||
no-descending-specificity: null
|
|
||||||
no-duplicate-at-import-rules: true
|
|
||||||
no-duplicate-selectors: true
|
|
||||||
no-empty-source: true
|
|
||||||
no-invalid-double-slash-comments: true
|
|
||||||
no-invalid-position-at-import-rule: [true, ignoreAtRules: [tailwind]]
|
|
||||||
no-irregular-whitespace: true
|
|
||||||
no-unknown-animations: null
|
|
||||||
no-unknown-custom-properties: null
|
|
||||||
number-max-precision: null
|
|
||||||
plugin/declaration-block-no-ignored-properties: true
|
|
||||||
property-allowed-list: null
|
|
||||||
property-disallowed-list: null
|
|
||||||
property-no-unknown: true
|
|
||||||
property-no-vendor-prefix: null
|
|
||||||
rule-empty-line-before: null
|
|
||||||
rule-selector-property-disallowed-list: null
|
|
||||||
scale-unlimited/declaration-strict-value: [[/color$/, font-weight], {ignoreValues: /^(inherit|transparent|unset|initial|currentcolor|none)$/, ignoreFunctions: false, disableFix: true, expandShorthand: true}]
|
|
||||||
selector-anb-no-unmatchable: true
|
|
||||||
selector-attribute-name-disallowed-list: null
|
|
||||||
selector-attribute-operator-allowed-list: null
|
|
||||||
selector-attribute-operator-disallowed-list: null
|
|
||||||
selector-attribute-quotes: always
|
|
||||||
selector-class-pattern: null
|
|
||||||
selector-combinator-allowed-list: null
|
|
||||||
selector-combinator-disallowed-list: null
|
|
||||||
selector-disallowed-list: null
|
|
||||||
selector-id-pattern: null
|
|
||||||
selector-max-attribute: null
|
|
||||||
selector-max-class: null
|
|
||||||
selector-max-combinators: null
|
|
||||||
selector-max-compound-selectors: null
|
|
||||||
selector-max-id: null
|
|
||||||
selector-max-pseudo-class: null
|
|
||||||
selector-max-specificity: null
|
|
||||||
selector-max-type: null
|
|
||||||
selector-max-universal: null
|
|
||||||
selector-nested-pattern: null
|
|
||||||
selector-no-qualifying-type: null
|
|
||||||
selector-no-vendor-prefix: true
|
|
||||||
selector-not-notation: null
|
|
||||||
selector-pseudo-class-allowed-list: null
|
|
||||||
selector-pseudo-class-disallowed-list: null
|
|
||||||
selector-pseudo-class-no-unknown: true
|
|
||||||
selector-pseudo-element-allowed-list: null
|
|
||||||
selector-pseudo-element-colon-notation: double
|
|
||||||
selector-pseudo-element-disallowed-list: null
|
|
||||||
selector-pseudo-element-no-unknown: true
|
|
||||||
selector-type-case: lower
|
|
||||||
selector-type-no-unknown: [true, {ignore: [custom-elements]}]
|
|
||||||
shorthand-property-no-redundant-values: true
|
|
||||||
string-no-newline: true
|
|
||||||
time-min-milliseconds: null
|
|
||||||
unit-allowed-list: null
|
|
||||||
unit-disallowed-list: null
|
|
||||||
unit-no-unknown: true
|
|
||||||
value-keyword-case: null
|
|
||||||
value-no-vendor-prefix: [true, {ignoreValues: [box, inline-box]}]
|
|
5223
CHANGELOG-archived.md
Normal file
5223
CHANGELOG-archived.md
Normal file
File diff suppressed because it is too large
Load Diff
5382
CHANGELOG.md
5382
CHANGELOG.md
File diff suppressed because it is too large
Load Diff
@ -77,7 +77,7 @@ If your issue has not been reported yet, [open an issue](https://github.com/go-g
|
|||||||
and answer the questions so we can understand and reproduce the problematic behavior. \
|
and answer the questions so we can understand and reproduce the problematic behavior. \
|
||||||
Please write clear and concise instructions so that we can reproduce the behavior — even if it seems obvious. \
|
Please write clear and concise instructions so that we can reproduce the behavior — even if it seems obvious. \
|
||||||
The more detailed and specific you are, the faster we can fix the issue. \
|
The more detailed and specific you are, the faster we can fix the issue. \
|
||||||
It is really helpful if you can reproduce your problem on a site running on the latest commits, i.e. <https://try.gitea.io>, as perhaps your problem has already been fixed on a current version. \
|
It is really helpful if you can reproduce your problem on a site running on the latest commits, i.e. <https://demo.gitea.com>, as perhaps your problem has already been fixed on a current version. \
|
||||||
Please follow the guidelines described in [How to Report Bugs Effectively](http://www.chiark.greenend.org.uk/~sgtatham/bugs.html) for your report.
|
Please follow the guidelines described in [How to Report Bugs Effectively](http://www.chiark.greenend.org.uk/~sgtatham/bugs.html) for your report.
|
||||||
|
|
||||||
Please be kind, remember that Gitea comes at no cost to you, and you're getting free help.
|
Please be kind, remember that Gitea comes at no cost to you, and you're getting free help.
|
||||||
@ -358,11 +358,12 @@ $REWRITTEN_PR_SUMMARY
|
|||||||
|
|
||||||
## Documentation
|
## Documentation
|
||||||
|
|
||||||
If you add a new feature or change an existing aspect of Gitea, the documentation for that feature must be created or updated in the same PR.
|
If you add a new feature or change an existing aspect of Gitea, the documentation for that feature must be created or updated in another PR at [https://gitea.com/gitea/docs](https://gitea.com/gitea/docs).
|
||||||
|
**The docs directory on main repository will be removed at some time. We will have a yaml file to store configuration file's meta data. After that completed, configuration documentation should be in the main repository.**
|
||||||
|
|
||||||
## API v1
|
## API v1
|
||||||
|
|
||||||
The API is documented by [swagger](http://try.gitea.io/api/swagger) and is based on [the GitHub API](https://docs.github.com/en/rest).
|
The API is documented by [swagger](https://gitea.com/api/swagger) and is based on [the GitHub API](https://docs.github.com/en/rest).
|
||||||
|
|
||||||
### GitHub API compatibility
|
### GitHub API compatibility
|
||||||
|
|
||||||
|
12
Dockerfile
12
Dockerfile
@ -1,12 +1,12 @@
|
|||||||
# Build stage
|
# Build stage
|
||||||
FROM docker.io/library/golang:1.22-alpine3.19 AS build-env
|
FROM docker.io/library/golang:1.22-alpine3.20 AS build-env
|
||||||
|
|
||||||
ARG GOPROXY
|
ARG GOPROXY
|
||||||
ENV GOPROXY ${GOPROXY:-direct}
|
ENV GOPROXY=${GOPROXY:-direct}
|
||||||
|
|
||||||
ARG GITEA_VERSION
|
ARG GITEA_VERSION
|
||||||
ARG TAGS="sqlite sqlite_unlock_notify"
|
ARG TAGS="sqlite sqlite_unlock_notify"
|
||||||
ENV TAGS "bindata timetzdata $TAGS"
|
ENV TAGS="bindata timetzdata $TAGS"
|
||||||
ARG CGO_EXTRA_CFLAGS
|
ARG CGO_EXTRA_CFLAGS
|
||||||
|
|
||||||
# Build deps
|
# Build deps
|
||||||
@ -41,7 +41,7 @@ RUN chmod 755 /tmp/local/usr/bin/entrypoint \
|
|||||||
/go/src/code.gitea.io/gitea/environment-to-ini
|
/go/src/code.gitea.io/gitea/environment-to-ini
|
||||||
RUN chmod 644 /go/src/code.gitea.io/gitea/contrib/autocompletion/bash_autocomplete
|
RUN chmod 644 /go/src/code.gitea.io/gitea/contrib/autocompletion/bash_autocomplete
|
||||||
|
|
||||||
FROM docker.io/library/alpine:3.19
|
FROM docker.io/library/alpine:3.20
|
||||||
LABEL maintainer="maintainers@gitea.io"
|
LABEL maintainer="maintainers@gitea.io"
|
||||||
|
|
||||||
EXPOSE 22 3000
|
EXPOSE 22 3000
|
||||||
@ -72,8 +72,8 @@ RUN addgroup \
|
|||||||
git && \
|
git && \
|
||||||
echo "git:*" | chpasswd -e
|
echo "git:*" | chpasswd -e
|
||||||
|
|
||||||
ENV USER git
|
ENV USER=git
|
||||||
ENV GITEA_CUSTOM /data/gitea
|
ENV GITEA_CUSTOM=/data/gitea
|
||||||
|
|
||||||
VOLUME ["/data"]
|
VOLUME ["/data"]
|
||||||
|
|
||||||
|
@ -1,12 +1,12 @@
|
|||||||
# Build stage
|
# Build stage
|
||||||
FROM docker.io/library/golang:1.22-alpine3.19 AS build-env
|
FROM docker.io/library/golang:1.22-alpine3.20 AS build-env
|
||||||
|
|
||||||
ARG GOPROXY
|
ARG GOPROXY
|
||||||
ENV GOPROXY ${GOPROXY:-direct}
|
ENV GOPROXY=${GOPROXY:-direct}
|
||||||
|
|
||||||
ARG GITEA_VERSION
|
ARG GITEA_VERSION
|
||||||
ARG TAGS="sqlite sqlite_unlock_notify"
|
ARG TAGS="sqlite sqlite_unlock_notify"
|
||||||
ENV TAGS "bindata timetzdata $TAGS"
|
ENV TAGS="bindata timetzdata $TAGS"
|
||||||
ARG CGO_EXTRA_CFLAGS
|
ARG CGO_EXTRA_CFLAGS
|
||||||
|
|
||||||
#Build deps
|
#Build deps
|
||||||
@ -39,7 +39,7 @@ RUN chmod 755 /tmp/local/usr/local/bin/docker-entrypoint.sh \
|
|||||||
/go/src/code.gitea.io/gitea/environment-to-ini
|
/go/src/code.gitea.io/gitea/environment-to-ini
|
||||||
RUN chmod 644 /go/src/code.gitea.io/gitea/contrib/autocompletion/bash_autocomplete
|
RUN chmod 644 /go/src/code.gitea.io/gitea/contrib/autocompletion/bash_autocomplete
|
||||||
|
|
||||||
FROM docker.io/library/alpine:3.19
|
FROM docker.io/library/alpine:3.20
|
||||||
LABEL maintainer="maintainers@gitea.io"
|
LABEL maintainer="maintainers@gitea.io"
|
||||||
|
|
||||||
EXPOSE 2222 3000
|
EXPOSE 2222 3000
|
||||||
@ -75,14 +75,14 @@ COPY --from=build-env /go/src/code.gitea.io/gitea/contrib/autocompletion/bash_au
|
|||||||
|
|
||||||
# git:git
|
# git:git
|
||||||
USER 1000:1000
|
USER 1000:1000
|
||||||
ENV GITEA_WORK_DIR /var/lib/gitea
|
ENV GITEA_WORK_DIR=/var/lib/gitea
|
||||||
ENV GITEA_CUSTOM /var/lib/gitea/custom
|
ENV GITEA_CUSTOM=/var/lib/gitea/custom
|
||||||
ENV GITEA_TEMP /tmp/gitea
|
ENV GITEA_TEMP=/tmp/gitea
|
||||||
ENV TMPDIR /tmp/gitea
|
ENV TMPDIR=/tmp/gitea
|
||||||
|
|
||||||
# TODO add to docs the ability to define the ini to load (useful to test and revert a config)
|
# TODO add to docs the ability to define the ini to load (useful to test and revert a config)
|
||||||
ENV GITEA_APP_INI /etc/gitea/app.ini
|
ENV GITEA_APP_INI=/etc/gitea/app.ini
|
||||||
ENV HOME "/var/lib/gitea/git"
|
ENV HOME="/var/lib/gitea/git"
|
||||||
VOLUME ["/var/lib/gitea", "/etc/gitea"]
|
VOLUME ["/var/lib/gitea", "/etc/gitea"]
|
||||||
WORKDIR /var/lib/gitea
|
WORKDIR /var/lib/gitea
|
||||||
|
|
||||||
|
@ -60,3 +60,6 @@ Nanguan Lin <nanguanlin6@gmail.com> (@lng2020)
|
|||||||
kerwin612 <kerwin612@qq.com> (@kerwin612)
|
kerwin612 <kerwin612@qq.com> (@kerwin612)
|
||||||
Gary Wang <git@blumia.net> (@BLumia)
|
Gary Wang <git@blumia.net> (@BLumia)
|
||||||
Tim-Niclas Oelschläger <zokki.softwareschmiede@gmail.com> (@zokkis)
|
Tim-Niclas Oelschläger <zokki.softwareschmiede@gmail.com> (@zokkis)
|
||||||
|
Yu Liu <1240335630@qq.com> (@HEREYUA)
|
||||||
|
Kemal Zebari <kemalzebra@gmail.com> (@kemzeb)
|
||||||
|
Rowan Bohde <rowan.bohde@gmail.com> (@bohde)
|
||||||
|
98
Makefile
98
Makefile
@ -25,17 +25,18 @@ COMMA := ,
|
|||||||
|
|
||||||
XGO_VERSION := go-1.22.x
|
XGO_VERSION := go-1.22.x
|
||||||
|
|
||||||
AIR_PACKAGE ?= github.com/cosmtrek/air@v1.49.0
|
AIR_PACKAGE ?= github.com/air-verse/air@v1
|
||||||
EDITORCONFIG_CHECKER_PACKAGE ?= github.com/editorconfig-checker/editorconfig-checker/cmd/editorconfig-checker@2.7.0
|
EDITORCONFIG_CHECKER_PACKAGE ?= github.com/editorconfig-checker/editorconfig-checker/cmd/editorconfig-checker@2.7.0
|
||||||
GOFUMPT_PACKAGE ?= mvdan.cc/gofumpt@v0.6.0
|
GOFUMPT_PACKAGE ?= mvdan.cc/gofumpt@v0.6.0
|
||||||
GOLANGCI_LINT_PACKAGE ?= github.com/golangci/golangci-lint/cmd/golangci-lint@v1.56.1
|
GOLANGCI_LINT_PACKAGE ?= github.com/golangci/golangci-lint/cmd/golangci-lint@v1.59.0
|
||||||
GXZ_PACKAGE ?= github.com/ulikunitz/xz/cmd/gxz@v0.5.11
|
GXZ_PACKAGE ?= github.com/ulikunitz/xz/cmd/gxz@v0.5.11
|
||||||
MISSPELL_PACKAGE ?= github.com/golangci/misspell/cmd/misspell@v0.4.1
|
MISSPELL_PACKAGE ?= github.com/golangci/misspell/cmd/misspell@v0.5.1
|
||||||
SWAGGER_PACKAGE ?= github.com/go-swagger/go-swagger/cmd/swagger@db51e79a0e37c572d8b59ae0c58bf2bbbbe53285
|
SWAGGER_PACKAGE ?= github.com/go-swagger/go-swagger/cmd/swagger@v0.31.0
|
||||||
XGO_PACKAGE ?= src.techknowlogick.com/xgo@latest
|
XGO_PACKAGE ?= src.techknowlogick.com/xgo@latest
|
||||||
GO_LICENSES_PACKAGE ?= github.com/google/go-licenses@v1.6.0
|
GO_LICENSES_PACKAGE ?= github.com/google/go-licenses@v1
|
||||||
GOVULNCHECK_PACKAGE ?= golang.org/x/vuln/cmd/govulncheck@v1.0.3
|
GOVULNCHECK_PACKAGE ?= golang.org/x/vuln/cmd/govulncheck@v1
|
||||||
ACTIONLINT_PACKAGE ?= github.com/rhysd/actionlint/cmd/actionlint@v1.6.26
|
ACTIONLINT_PACKAGE ?= github.com/rhysd/actionlint/cmd/actionlint@v1
|
||||||
|
GOPLS_PACKAGE ?= golang.org/x/tools/gopls@v0.15.3
|
||||||
|
|
||||||
DOCKER_IMAGE ?= gitea/gitea
|
DOCKER_IMAGE ?= gitea/gitea
|
||||||
DOCKER_TAG ?= latest
|
DOCKER_TAG ?= latest
|
||||||
@ -88,7 +89,7 @@ ifneq ($(GITHUB_REF_TYPE),branch)
|
|||||||
GITEA_VERSION ?= $(VERSION)
|
GITEA_VERSION ?= $(VERSION)
|
||||||
else
|
else
|
||||||
ifneq ($(GITHUB_REF_NAME),)
|
ifneq ($(GITHUB_REF_NAME),)
|
||||||
VERSION ?= $(subst release/v,,$(GITHUB_REF_NAME))
|
VERSION ?= $(subst release/v,,$(GITHUB_REF_NAME))-nightly
|
||||||
else
|
else
|
||||||
VERSION ?= main
|
VERSION ?= main
|
||||||
endif
|
endif
|
||||||
@ -110,7 +111,6 @@ LDFLAGS := $(LDFLAGS) -X "main.MakeVersion=$(MAKE_VERSION)" -X "main.Version=$(G
|
|||||||
|
|
||||||
LINUX_ARCHS ?= linux/amd64,linux/386,linux/arm-5,linux/arm-6,linux/arm64
|
LINUX_ARCHS ?= linux/amd64,linux/386,linux/arm-5,linux/arm-6,linux/arm64
|
||||||
|
|
||||||
GO_PACKAGES ?= $(filter-out code.gitea.io/gitea/tests/integration/migration-test code.gitea.io/gitea/tests code.gitea.io/gitea/tests/integration code.gitea.io/gitea/tests/e2e,$(shell $(GO) list ./... | grep -v /vendor/))
|
|
||||||
GO_TEST_PACKAGES ?= $(filter-out $(shell $(GO) list code.gitea.io/gitea/models/migrations/...) code.gitea.io/gitea/tests/integration/migration-test code.gitea.io/gitea/tests code.gitea.io/gitea/tests/integration code.gitea.io/gitea/tests/e2e,$(shell $(GO) list ./... | grep -v /vendor/))
|
GO_TEST_PACKAGES ?= $(filter-out $(shell $(GO) list code.gitea.io/gitea/models/migrations/...) code.gitea.io/gitea/tests/integration/migration-test code.gitea.io/gitea/tests code.gitea.io/gitea/tests/integration code.gitea.io/gitea/tests/e2e,$(shell $(GO) list ./... | grep -v /vendor/))
|
||||||
MIGRATE_TEST_PACKAGES ?= $(shell $(GO) list code.gitea.io/gitea/models/migrations/...)
|
MIGRATE_TEST_PACKAGES ?= $(shell $(GO) list code.gitea.io/gitea/models/migrations/...)
|
||||||
|
|
||||||
@ -119,7 +119,7 @@ FOMANTIC_WORK_DIR := web_src/fomantic
|
|||||||
WEBPACK_SOURCES := $(shell find web_src/js web_src/css -type f)
|
WEBPACK_SOURCES := $(shell find web_src/js web_src/css -type f)
|
||||||
WEBPACK_CONFIGS := webpack.config.js tailwind.config.js
|
WEBPACK_CONFIGS := webpack.config.js tailwind.config.js
|
||||||
WEBPACK_DEST := public/assets/js/index.js public/assets/css/index.css
|
WEBPACK_DEST := public/assets/js/index.js public/assets/css/index.css
|
||||||
WEBPACK_DEST_ENTRIES := public/assets/js public/assets/css public/assets/fonts public/assets/img/webpack
|
WEBPACK_DEST_ENTRIES := public/assets/js public/assets/css public/assets/fonts
|
||||||
|
|
||||||
BINDATA_DEST := modules/public/bindata.go modules/options/bindata.go modules/templates/bindata.go
|
BINDATA_DEST := modules/public/bindata.go modules/options/bindata.go modules/templates/bindata.go
|
||||||
BINDATA_HASH := $(addsuffix .hash,$(BINDATA_DEST))
|
BINDATA_HASH := $(addsuffix .hash,$(BINDATA_DEST))
|
||||||
@ -137,16 +137,16 @@ TAGS ?=
|
|||||||
TAGS_SPLIT := $(subst $(COMMA), ,$(TAGS))
|
TAGS_SPLIT := $(subst $(COMMA), ,$(TAGS))
|
||||||
TAGS_EVIDENCE := $(MAKE_EVIDENCE_DIR)/tags
|
TAGS_EVIDENCE := $(MAKE_EVIDENCE_DIR)/tags
|
||||||
|
|
||||||
TEST_TAGS ?= sqlite sqlite_unlock_notify
|
TEST_TAGS ?= $(TAGS_SPLIT) sqlite sqlite_unlock_notify
|
||||||
|
|
||||||
TAR_EXCLUDES := .git data indexers queues log node_modules $(EXECUTABLE) $(FOMANTIC_WORK_DIR)/node_modules $(DIST) $(MAKE_EVIDENCE_DIR) $(AIR_TMP_DIR) $(GO_LICENSE_TMP_DIR)
|
TAR_EXCLUDES := .git data indexers queues log node_modules $(EXECUTABLE) $(FOMANTIC_WORK_DIR)/node_modules $(DIST) $(MAKE_EVIDENCE_DIR) $(AIR_TMP_DIR) $(GO_LICENSE_TMP_DIR)
|
||||||
|
|
||||||
GO_DIRS := build cmd models modules routers services tests
|
GO_DIRS := build cmd models modules routers services tests
|
||||||
WEB_DIRS := web_src/js web_src/css
|
WEB_DIRS := web_src/js web_src/css
|
||||||
|
|
||||||
ESLINT_FILES := web_src/js tools *.config.js tests/e2e
|
ESLINT_FILES := web_src/js tools *.js *.ts tests/e2e
|
||||||
STYLELINT_FILES := web_src/css web_src/js/components/*.vue
|
STYLELINT_FILES := web_src/css web_src/js/components/*.vue
|
||||||
SPELLCHECK_FILES := $(GO_DIRS) $(WEB_DIRS) docs/content templates options/locale/locale_en-US.ini .github
|
SPELLCHECK_FILES := $(GO_DIRS) $(WEB_DIRS) templates options/locale/locale_en-US.ini .github $(filter-out CHANGELOG.md, $(wildcard *.go *.js *.md *.yml *.yaml *.toml))
|
||||||
EDITORCONFIG_FILES := templates .github/workflows options/locale/locale_en-US.ini
|
EDITORCONFIG_FILES := templates .github/workflows options/locale/locale_en-US.ini
|
||||||
|
|
||||||
GO_SOURCES := $(wildcard *.go)
|
GO_SOURCES := $(wildcard *.go)
|
||||||
@ -214,6 +214,7 @@ help:
|
|||||||
@echo " - lint-go lint go files"
|
@echo " - lint-go lint go files"
|
||||||
@echo " - lint-go-fix lint go files and fix issues"
|
@echo " - lint-go-fix lint go files and fix issues"
|
||||||
@echo " - lint-go-vet lint go files with vet"
|
@echo " - lint-go-vet lint go files with vet"
|
||||||
|
@echo " - lint-go-gopls lint go files with gopls"
|
||||||
@echo " - lint-js lint js files"
|
@echo " - lint-js lint js files"
|
||||||
@echo " - lint-js-fix lint js files and fix issues"
|
@echo " - lint-js-fix lint js files and fix issues"
|
||||||
@echo " - lint-css lint css files"
|
@echo " - lint-css lint css files"
|
||||||
@ -295,7 +296,7 @@ clean:
|
|||||||
|
|
||||||
.PHONY: fmt
|
.PHONY: fmt
|
||||||
fmt:
|
fmt:
|
||||||
GOFUMPT_PACKAGE=$(GOFUMPT_PACKAGE) $(GO) run build/code-batch-process.go gitea-fmt -w '{file-list}'
|
@GOFUMPT_PACKAGE=$(GOFUMPT_PACKAGE) $(GO) run build/code-batch-process.go gitea-fmt -w '{file-list}'
|
||||||
$(eval TEMPLATES := $(shell find templates -type f -name '*.tmpl'))
|
$(eval TEMPLATES := $(shell find templates -type f -name '*.tmpl'))
|
||||||
@# strip whitespace after '{{' or '(' and before '}}' or ')' unless there is only
|
@# strip whitespace after '{{' or '(' and before '}}' or ')' unless there is only
|
||||||
@# whitespace before it
|
@# whitespace before it
|
||||||
@ -367,18 +368,20 @@ lint-frontend: lint-js lint-css
|
|||||||
lint-frontend-fix: lint-js-fix lint-css-fix
|
lint-frontend-fix: lint-js-fix lint-css-fix
|
||||||
|
|
||||||
.PHONY: lint-backend
|
.PHONY: lint-backend
|
||||||
lint-backend: lint-go lint-go-vet lint-editorconfig
|
lint-backend: lint-go lint-go-vet lint-go-gopls lint-editorconfig
|
||||||
|
|
||||||
.PHONY: lint-backend-fix
|
.PHONY: lint-backend-fix
|
||||||
lint-backend-fix: lint-go-fix lint-go-vet lint-editorconfig
|
lint-backend-fix: lint-go-fix lint-go-vet lint-editorconfig
|
||||||
|
|
||||||
.PHONY: lint-js
|
.PHONY: lint-js
|
||||||
lint-js: node_modules
|
lint-js: node_modules
|
||||||
npx eslint --color --max-warnings=0 --ext js,vue $(ESLINT_FILES)
|
npx eslint --color --max-warnings=0 --ext js,ts,vue $(ESLINT_FILES)
|
||||||
|
# npx tsc
|
||||||
|
|
||||||
.PHONY: lint-js-fix
|
.PHONY: lint-js-fix
|
||||||
lint-js-fix: node_modules
|
lint-js-fix: node_modules
|
||||||
npx eslint --color --max-warnings=0 --ext js,vue $(ESLINT_FILES) --fix
|
npx eslint --color --max-warnings=0 --ext js,ts,vue $(ESLINT_FILES) --fix
|
||||||
|
# npx tsc
|
||||||
|
|
||||||
.PHONY: lint-css
|
.PHONY: lint-css
|
||||||
lint-css: node_modules
|
lint-css: node_modules
|
||||||
@ -394,15 +397,15 @@ lint-swagger: node_modules
|
|||||||
|
|
||||||
.PHONY: lint-md
|
.PHONY: lint-md
|
||||||
lint-md: node_modules
|
lint-md: node_modules
|
||||||
npx markdownlint docs *.md
|
npx markdownlint *.md
|
||||||
|
|
||||||
.PHONY: lint-spell
|
.PHONY: lint-spell
|
||||||
lint-spell:
|
lint-spell:
|
||||||
@go run $(MISSPELL_PACKAGE) -error $(SPELLCHECK_FILES)
|
@go run $(MISSPELL_PACKAGE) -dict tools/misspellings.csv -error $(SPELLCHECK_FILES)
|
||||||
|
|
||||||
.PHONY: lint-spell-fix
|
.PHONY: lint-spell-fix
|
||||||
lint-spell-fix:
|
lint-spell-fix:
|
||||||
@go run $(MISSPELL_PACKAGE) -w $(SPELLCHECK_FILES)
|
@go run $(MISSPELL_PACKAGE) -dict tools/misspellings.csv -w $(SPELLCHECK_FILES)
|
||||||
|
|
||||||
.PHONY: lint-go
|
.PHONY: lint-go
|
||||||
lint-go:
|
lint-go:
|
||||||
@ -423,7 +426,12 @@ lint-go-windows:
|
|||||||
lint-go-vet:
|
lint-go-vet:
|
||||||
@echo "Running go vet..."
|
@echo "Running go vet..."
|
||||||
@GOOS= GOARCH= $(GO) build code.gitea.io/gitea-vet
|
@GOOS= GOARCH= $(GO) build code.gitea.io/gitea-vet
|
||||||
@$(GO) vet -vettool=gitea-vet $(GO_PACKAGES)
|
@$(GO) vet -vettool=gitea-vet ./...
|
||||||
|
|
||||||
|
.PHONY: lint-go-gopls
|
||||||
|
lint-go-gopls:
|
||||||
|
@echo "Running gopls check..."
|
||||||
|
@GO=$(GO) GOPLS_PACKAGE=$(GOPLS_PACKAGE) tools/lint-go-gopls.sh $(GO_SOURCES_NO_BINDATA)
|
||||||
|
|
||||||
.PHONY: lint-editorconfig
|
.PHONY: lint-editorconfig
|
||||||
lint-editorconfig:
|
lint-editorconfig:
|
||||||
@ -434,7 +442,8 @@ lint-actions:
|
|||||||
$(GO) run $(ACTIONLINT_PACKAGE)
|
$(GO) run $(ACTIONLINT_PACKAGE)
|
||||||
|
|
||||||
.PHONY: lint-templates
|
.PHONY: lint-templates
|
||||||
lint-templates: .venv
|
lint-templates: .venv node_modules
|
||||||
|
@node tools/lint-templates-svg.js
|
||||||
@poetry run djlint $(shell find templates -type f -iname '*.tmpl')
|
@poetry run djlint $(shell find templates -type f -iname '*.tmpl')
|
||||||
|
|
||||||
.PHONY: lint-yaml
|
.PHONY: lint-yaml
|
||||||
@ -778,7 +787,7 @@ generate-backend: $(TAGS_PREREQ) generate-go
|
|||||||
.PHONY: generate-go
|
.PHONY: generate-go
|
||||||
generate-go: $(TAGS_PREREQ)
|
generate-go: $(TAGS_PREREQ)
|
||||||
@echo "Running go generate..."
|
@echo "Running go generate..."
|
||||||
@CC= GOOS= GOARCH= $(GO) generate -tags '$(TAGS)' $(GO_PACKAGES)
|
@CC= GOOS= GOARCH= CGO_ENABLED=0 $(GO) generate -tags '$(TAGS)' ./...
|
||||||
|
|
||||||
.PHONY: security-check
|
.PHONY: security-check
|
||||||
security-check:
|
security-check:
|
||||||
@ -788,7 +797,7 @@ $(EXECUTABLE): $(GO_SOURCES) $(TAGS_PREREQ)
|
|||||||
CGO_CFLAGS="$(CGO_CFLAGS)" $(GO) build $(GOFLAGS) $(EXTRA_GOFLAGS) -tags '$(TAGS)' -ldflags '-s -w $(LDFLAGS)' -o $@
|
CGO_CFLAGS="$(CGO_CFLAGS)" $(GO) build $(GOFLAGS) $(EXTRA_GOFLAGS) -tags '$(TAGS)' -ldflags '-s -w $(LDFLAGS)' -o $@
|
||||||
|
|
||||||
.PHONY: release
|
.PHONY: release
|
||||||
release: frontend generate release-windows release-linux release-darwin release-freebsd release-copy release-compress vendor release-sources release-docs release-check
|
release: frontend generate release-windows release-linux release-darwin release-freebsd release-copy release-compress vendor release-sources release-check
|
||||||
|
|
||||||
$(DIST_DIRS):
|
$(DIST_DIRS):
|
||||||
mkdir -p $(DIST_DIRS)
|
mkdir -p $(DIST_DIRS)
|
||||||
@ -834,10 +843,6 @@ release-sources: | $(DIST_DIRS)
|
|||||||
tar $(addprefix $(EXCL),$(TAR_EXCLUDES)) $(TRANSFORM) -czf $(DIST)/release/gitea-src-$(VERSION).tar.gz .
|
tar $(addprefix $(EXCL),$(TAR_EXCLUDES)) $(TRANSFORM) -czf $(DIST)/release/gitea-src-$(VERSION).tar.gz .
|
||||||
rm -f $(STORED_VERSION_FILE)
|
rm -f $(STORED_VERSION_FILE)
|
||||||
|
|
||||||
.PHONY: release-docs
|
|
||||||
release-docs: | $(DIST_DIRS) docs
|
|
||||||
tar -czf $(DIST)/release/gitea-docs-$(VERSION).tar.gz -C ./docs .
|
|
||||||
|
|
||||||
.PHONY: deps
|
.PHONY: deps
|
||||||
deps: deps-frontend deps-backend deps-tools deps-py
|
deps: deps-frontend deps-backend deps-tools deps-py
|
||||||
|
|
||||||
@ -853,24 +858,26 @@ deps-backend:
|
|||||||
|
|
||||||
.PHONY: deps-tools
|
.PHONY: deps-tools
|
||||||
deps-tools:
|
deps-tools:
|
||||||
$(GO) install $(AIR_PACKAGE)
|
$(GO) install $(AIR_PACKAGE) & \
|
||||||
$(GO) install $(EDITORCONFIG_CHECKER_PACKAGE)
|
$(GO) install $(EDITORCONFIG_CHECKER_PACKAGE) & \
|
||||||
$(GO) install $(GOFUMPT_PACKAGE)
|
$(GO) install $(GOFUMPT_PACKAGE) & \
|
||||||
$(GO) install $(GOLANGCI_LINT_PACKAGE)
|
$(GO) install $(GOLANGCI_LINT_PACKAGE) & \
|
||||||
$(GO) install $(GXZ_PACKAGE)
|
$(GO) install $(GXZ_PACKAGE) & \
|
||||||
$(GO) install $(MISSPELL_PACKAGE)
|
$(GO) install $(MISSPELL_PACKAGE) & \
|
||||||
$(GO) install $(SWAGGER_PACKAGE)
|
$(GO) install $(SWAGGER_PACKAGE) & \
|
||||||
$(GO) install $(XGO_PACKAGE)
|
$(GO) install $(XGO_PACKAGE) & \
|
||||||
$(GO) install $(GO_LICENSES_PACKAGE)
|
$(GO) install $(GO_LICENSES_PACKAGE) & \
|
||||||
$(GO) install $(GOVULNCHECK_PACKAGE)
|
$(GO) install $(GOVULNCHECK_PACKAGE) & \
|
||||||
$(GO) install $(ACTIONLINT_PACKAGE)
|
$(GO) install $(ACTIONLINT_PACKAGE) & \
|
||||||
|
$(GO) install $(GOPLS_PACKAGE) & \
|
||||||
|
wait
|
||||||
|
|
||||||
node_modules: package-lock.json
|
node_modules: package-lock.json
|
||||||
npm install --no-save
|
npm install --no-save
|
||||||
@touch node_modules
|
@touch node_modules
|
||||||
|
|
||||||
.venv: poetry.lock
|
.venv: poetry.lock
|
||||||
poetry install --no-root
|
poetry install
|
||||||
@touch .venv
|
@touch .venv
|
||||||
|
|
||||||
.PHONY: update
|
.PHONY: update
|
||||||
@ -881,13 +888,15 @@ update-js: node-check | node_modules
|
|||||||
npx updates -u -f package.json
|
npx updates -u -f package.json
|
||||||
rm -rf node_modules package-lock.json
|
rm -rf node_modules package-lock.json
|
||||||
npm install --package-lock
|
npm install --package-lock
|
||||||
|
npx nolyfill install
|
||||||
|
npm install --package-lock
|
||||||
@touch node_modules
|
@touch node_modules
|
||||||
|
|
||||||
.PHONY: update-py
|
.PHONY: update-py
|
||||||
update-py: node-check | node_modules
|
update-py: node-check | node_modules
|
||||||
npx updates -u -f pyproject.toml
|
npx updates -u -f pyproject.toml
|
||||||
rm -rf .venv poetry.lock
|
rm -rf .venv poetry.lock
|
||||||
poetry install --no-root
|
poetry install
|
||||||
@touch .venv
|
@touch .venv
|
||||||
|
|
||||||
.PHONY: fomantic
|
.PHONY: fomantic
|
||||||
@ -908,8 +917,9 @@ webpack: $(WEBPACK_DEST)
|
|||||||
|
|
||||||
$(WEBPACK_DEST): $(WEBPACK_SOURCES) $(WEBPACK_CONFIGS) package-lock.json
|
$(WEBPACK_DEST): $(WEBPACK_SOURCES) $(WEBPACK_CONFIGS) package-lock.json
|
||||||
@$(MAKE) -s node-check node_modules
|
@$(MAKE) -s node-check node_modules
|
||||||
rm -rf $(WEBPACK_DEST_ENTRIES)
|
@rm -rf $(WEBPACK_DEST_ENTRIES)
|
||||||
npx webpack
|
@echo "Running webpack..."
|
||||||
|
@BROWSERSLIST_IGNORE_OLD_DATA=true npx webpack
|
||||||
@touch $(WEBPACK_DEST)
|
@touch $(WEBPACK_DEST)
|
||||||
|
|
||||||
.PHONY: svg
|
.PHONY: svg
|
||||||
@ -958,7 +968,7 @@ generate-gitignore:
|
|||||||
|
|
||||||
.PHONY: generate-images
|
.PHONY: generate-images
|
||||||
generate-images: | node_modules
|
generate-images: | node_modules
|
||||||
npm install --no-save fabric@6.0.0-beta19 imagemin-zopfli@7
|
npm install --no-save fabric@6 imagemin-zopfli@7
|
||||||
node tools/generate-images.js $(TAGS)
|
node tools/generate-images.js $(TAGS)
|
||||||
|
|
||||||
.PHONY: generate-manpage
|
.PHONY: generate-manpage
|
||||||
|
@ -8,9 +8,8 @@
|
|||||||
[![](https://www.codetriage.com/go-gitea/gitea/badges/users.svg)](https://www.codetriage.com/go-gitea/gitea "Help Contribute to Open Source")
|
[![](https://www.codetriage.com/go-gitea/gitea/badges/users.svg)](https://www.codetriage.com/go-gitea/gitea "Help Contribute to Open Source")
|
||||||
[![](https://opencollective.com/gitea/tiers/backers/badge.svg?label=backers&color=brightgreen)](https://opencollective.com/gitea "Become a backer/sponsor of gitea")
|
[![](https://opencollective.com/gitea/tiers/backers/badge.svg?label=backers&color=brightgreen)](https://opencollective.com/gitea "Become a backer/sponsor of gitea")
|
||||||
[![](https://img.shields.io/badge/License-MIT-blue.svg)](https://opensource.org/licenses/MIT "License: MIT")
|
[![](https://img.shields.io/badge/License-MIT-blue.svg)](https://opensource.org/licenses/MIT "License: MIT")
|
||||||
[![Contribute with Gitpod](https://img.shields.io/badge/Contribute%20with-Gitpod-908a85?logo=gitpod)](https://gitpod.io/#https://github.com/go-gitea/gitea)
|
[![Contribute with Gitpod](https://img.shields.io/badge/Contribute%20with-Gitpod-908a85?logo=gitpod&color=green)](https://gitpod.io/#https://github.com/go-gitea/gitea)
|
||||||
[![](https://badges.crowdin.net/gitea/localized.svg)](https://crowdin.com/project/gitea "Crowdin")
|
[![](https://badges.crowdin.net/gitea/localized.svg)](https://crowdin.com/project/gitea "Crowdin")
|
||||||
[![](https://badgen.net/https/api.tickgit.com/badgen/github.com/go-gitea/gitea/main)](https://www.tickgit.com/browse?repo=github.com/go-gitea/gitea&branch=main "TODOs")
|
|
||||||
|
|
||||||
[View this document in Chinese](./README_ZH.md)
|
[View this document in Chinese](./README_ZH.md)
|
||||||
|
|
||||||
@ -26,7 +25,7 @@ This project has been
|
|||||||
[forked](https://blog.gitea.com/welcome-to-gitea/) from
|
[forked](https://blog.gitea.com/welcome-to-gitea/) from
|
||||||
[Gogs](https://gogs.io) since November of 2016, but a lot has changed.
|
[Gogs](https://gogs.io) since November of 2016, but a lot has changed.
|
||||||
|
|
||||||
For online demonstrations, you can visit [try.gitea.io](https://try.gitea.io).
|
For online demonstrations, you can visit [demo.gitea.com](https://demo.gitea.com).
|
||||||
|
|
||||||
For accessing free Gitea service (with a limited number of repositories), you can visit [gitea.com](https://gitea.com/user/login).
|
For accessing free Gitea service (with a limited number of repositories), you can visit [gitea.com](https://gitea.com/user/login).
|
||||||
|
|
||||||
@ -56,7 +55,7 @@ More info: https://docs.gitea.com/installation/install-from-source
|
|||||||
./gitea web
|
./gitea web
|
||||||
|
|
||||||
> [!NOTE]
|
> [!NOTE]
|
||||||
> If you're interested in using our APIs, we have experimental support with [documentation](https://try.gitea.io/api/swagger).
|
> If you're interested in using our APIs, we have experimental support with [documentation](https://docs.gitea.com/api).
|
||||||
|
|
||||||
## Contributing
|
## Contributing
|
||||||
|
|
||||||
@ -80,7 +79,7 @@ https://docs.gitea.com/contributing/localization
|
|||||||
## Further information
|
## Further information
|
||||||
|
|
||||||
For more information and instructions about how to install Gitea, please look at our [documentation](https://docs.gitea.com/).
|
For more information and instructions about how to install Gitea, please look at our [documentation](https://docs.gitea.com/).
|
||||||
If you have questions that are not covered by the documentation, you can get in contact with us on our [Discord server](https://discord.gg/Gitea) or create a post in the [discourse forum](https://discourse.gitea.io/).
|
If you have questions that are not covered by the documentation, you can get in contact with us on our [Discord server](https://discord.gg/Gitea) or create a post in the [discourse forum](https://forum.gitea.com/).
|
||||||
|
|
||||||
We maintain a list of Gitea-related projects at [gitea/awesome-gitea](https://gitea.com/gitea/awesome-gitea).
|
We maintain a list of Gitea-related projects at [gitea/awesome-gitea](https://gitea.com/gitea/awesome-gitea).
|
||||||
|
|
||||||
|
@ -8,9 +8,8 @@
|
|||||||
[![](https://www.codetriage.com/go-gitea/gitea/badges/users.svg)](https://www.codetriage.com/go-gitea/gitea "Help Contribute to Open Source")
|
[![](https://www.codetriage.com/go-gitea/gitea/badges/users.svg)](https://www.codetriage.com/go-gitea/gitea "Help Contribute to Open Source")
|
||||||
[![](https://opencollective.com/gitea/tiers/backers/badge.svg?label=backers&color=brightgreen)](https://opencollective.com/gitea "Become a backer/sponsor of gitea")
|
[![](https://opencollective.com/gitea/tiers/backers/badge.svg?label=backers&color=brightgreen)](https://opencollective.com/gitea "Become a backer/sponsor of gitea")
|
||||||
[![](https://img.shields.io/badge/License-MIT-blue.svg)](https://opensource.org/licenses/MIT "License: MIT")
|
[![](https://img.shields.io/badge/License-MIT-blue.svg)](https://opensource.org/licenses/MIT "License: MIT")
|
||||||
[![Contribute with Gitpod](https://img.shields.io/badge/Contribute%20with-Gitpod-908a85?logo=gitpod)](https://gitpod.io/#https://github.com/go-gitea/gitea)
|
[![Contribute with Gitpod](https://img.shields.io/badge/Contribute%20with-Gitpod-908a85?logo=gitpod&color=green)](https://gitpod.io/#https://github.com/go-gitea/gitea)
|
||||||
[![](https://badges.crowdin.net/gitea/localized.svg)](https://crowdin.com/project/gitea "Crowdin")
|
[![](https://badges.crowdin.net/gitea/localized.svg)](https://crowdin.com/project/gitea "Crowdin")
|
||||||
[![](https://badgen.net/https/api.tickgit.com/badgen/github.com/go-gitea/gitea/main)](https://www.tickgit.com/browse?repo=github.com/go-gitea/gitea&branch=main "TODOs")
|
|
||||||
|
|
||||||
[View this document in English](./README.md)
|
[View this document in English](./README.md)
|
||||||
|
|
||||||
@ -18,7 +17,7 @@
|
|||||||
|
|
||||||
Gitea 的首要目标是创建一个极易安装,运行非常快速,安装和使用体验良好的自建 Git 服务。我们采用 Go 作为后端语言,这使我们只要生成一个可执行程序即可。并且他还支持跨平台,支持 Linux, macOS 和 Windows 以及各种架构,除了 x86,amd64,还包括 ARM 和 PowerPC。
|
Gitea 的首要目标是创建一个极易安装,运行非常快速,安装和使用体验良好的自建 Git 服务。我们采用 Go 作为后端语言,这使我们只要生成一个可执行程序即可。并且他还支持跨平台,支持 Linux, macOS 和 Windows 以及各种架构,除了 x86,amd64,还包括 ARM 和 PowerPC。
|
||||||
|
|
||||||
如果你想试用在线演示,请访问 [try.gitea.io](https://try.gitea.io/)。
|
如果你想试用在线演示和报告问题,请访问 [demo.gitea.com](https://demo.gitea.com/)。
|
||||||
|
|
||||||
如果你想使用免费的 Gitea 服务(有仓库数量限制),请访问 [gitea.com](https://gitea.com/user/login)。
|
如果你想使用免费的 Gitea 服务(有仓库数量限制),请访问 [gitea.com](https://gitea.com/user/login)。
|
||||||
|
|
||||||
|
60
SECURITY.md
60
SECURITY.md
@ -14,12 +14,12 @@ Please **DO NOT** file a public issue, instead send your report privately to `se
|
|||||||
|
|
||||||
Due to the sensitive nature of security information, you can use the below GPG public key to encrypt your mail body.
|
Due to the sensitive nature of security information, you can use the below GPG public key to encrypt your mail body.
|
||||||
|
|
||||||
The PGP key is valid until June 24, 2024.
|
The PGP key is valid until July 9, 2025.
|
||||||
|
|
||||||
```
|
```
|
||||||
Key ID: 6FCD2D5B
|
Key ID: 6FCD2D5B
|
||||||
Key Type: RSA
|
Key Type: RSA
|
||||||
Expires: 6/24/2024
|
Expires: 7/9/2025
|
||||||
Key Size: 4096/4096
|
Key Size: 4096/4096
|
||||||
Fingerprint: 3DE0 3D1E 144A 7F06 9359 99DC AAFD 2381 6FCD 2D5B
|
Fingerprint: 3DE0 3D1E 144A 7F06 9359 99DC AAFD 2381 6FCD 2D5B
|
||||||
```
|
```
|
||||||
@ -40,20 +40,20 @@ q+pHZl24JYR0Kf3T/ZiOC0cGd2QJqpJtg5J6S/OqfX9NH6MsCczO8pUC1N/aHH2X
|
|||||||
CTme2nF56izORqDWKoiICteL3GpYsCV9nyCidcCmoQsS+DKvE86YhIhVIVWGRY2F
|
CTme2nF56izORqDWKoiICteL3GpYsCV9nyCidcCmoQsS+DKvE86YhIhVIVWGRY2F
|
||||||
lzpAjnN9/KLtQroutrm+Ft0mdjDiJUeFVl1cOHDhoyfCsQh62HumoyZoZvqzQd6e
|
lzpAjnN9/KLtQroutrm+Ft0mdjDiJUeFVl1cOHDhoyfCsQh62HumoyZoZvqzQd6e
|
||||||
AbN11nq6aViMe2Q3je1AbiBnRnQSHxt1Tc8X4IshO3MQK1Sk7oPI6LA5oQARAQAB
|
AbN11nq6aViMe2Q3je1AbiBnRnQSHxt1Tc8X4IshO3MQK1Sk7oPI6LA5oQARAQAB
|
||||||
tCJHaXRlYSBTZWN1cml0eSA8c2VjdXJpdHlAZ2l0ZWEuaW8+iQJXBBMBCABBFiEE
|
tCJHaXRlYSBTZWN1cml0eSA8c2VjdXJpdHlAZ2l0ZWEuaW8+iQJXBBMBCABBAhsD
|
||||||
PeA9HhRKfwaTWZncqv0jgW/NLVsFAmK1Z/4CGwMFCQPCZwAFCwkIBwICIgIGFQoJ
|
BQsJCAcCAiICBhUKCQgLAgQWAgMBAh4HAheAFiEEPeA9HhRKfwaTWZncqv0jgW/N
|
||||||
CAsCBBYCAwECHgcCF4AACgkQqv0jgW/NLVvnyxAAhxyNnWzw/rQO2qhzqicmZM94
|
LVsFAmaMse0FCQW4fW8ACgkQqv0jgW/NLVtXLg/+PF4G9Jhlui15BTNlEBJAV2P/
|
||||||
njSbOg+U2qMBvCdaqCQQeC+uaMmMzkDPanUUmLcyCkWqfCjPNjeSXAkE9npepVJI
|
1QlAV2krk0fP7tykn0FR9RfGIfVV/kwC1f+ouosYPQDDevl9LWdUIM+g94DtNo2o
|
||||||
4HtmgxZQ94OU/h3CLbft+9GVRzUkVI29TSYGdvNtV2/BkNGoFFnKWQr119um0o6A
|
7ACpcL3morvt5lVGpIZHL8TbX0qmFRXL/pB/cB+K6IwYvh2mrbp2zH+r4SCRyFYq
|
||||||
bgha2Uy5uY8o3ZIoiKkiHRaEoWIjjeBxJxYAojsZY4YElUmsQ3ik2joG6rhFesTa
|
BjgXYFTI1MylJ1ShAjU6Z+m3oJ+2xs5LzHS0X6zkTjzA2Zl4zQzciQ9T+wJcE7Zi
|
||||||
ofVt/bL8G2xzpOG26WGIxBbqf2qjV6OtZ0hu/vtTPHeIWMLq0Mz0V3PEDQWfkGPE
|
HXdM1+YMF8KGNP8J9Rpug5oNDJ98lgZirRY7c3A/1xmYBiPnULwuuymdqEZO7l70
|
||||||
i2RYxxYDs2xzJhSQWqTNVLSq0m5xTJnbHhQPfdCX4C2jvFKgLdfmytQq49S7jiJb
|
SeAlE1RWYX8kbOBnBb/KY4XwE3Vic1oEzc9DiPWVH1ElX86WNNsFzuyULiwoBoWg
|
||||||
Z03HVOZ/PsyBlQfH9xJi06R5yQCMEA8h8Z5r3/NXW09kQ6OFRe6xshoTcxZGRPTo
|
pqZGhL9x1p5+46RGQSDczsHM7YGVtfYOiDo2PAVrmwsT0BnXnK8Oe3YIkvmUPEJu
|
||||||
srhwr3uPbmCRh+YEl7qBLU6+BC5k8IRTZXqhrj/aPJu3MxgbgwV8u3vLoFSXM2lb
|
OkLt0Z6A5n8pz8zhQzuApwBsK4ncJ8zTCpvz/pfKKqZC/Vnoh3gKGhDGvOZ+b5IJ
|
||||||
a61FgeCQ0O7lkgVswwF0RppCaH9Ul3ZDapet/vCRg4NVwm9zOI/8q/Vj0FKA1GDR
|
0kUTe2JsbnwFixDUMDtacQ1op8XOyLoLVmgqLn0+Pws4XPBlMof2bioFir3yHKnP
|
||||||
JhRu8+Ce8zlFL65D34t+PprAzSeTlbv9um3x/ZIjCco7EEKSBylt+AZj/VyA6+e5
|
gNchsF1agrlSIo5GA8u4ga+IlCSfvFIKrl7+cxacKcJYt/vbOU5KcvVJI5EtHKCG
|
||||||
kjOQwRRc6dFJWBcorsSI2dG+H+QMF7ZabzmeCcz1v9HjLOPzYHoZAHhCmSppWTvX
|
xfHjHY2ah1Qww7SxW6IXiRZZzPpsL2mBM2CD7N3qh9bV2s27wxYCdUodsIZbiyHe
|
||||||
AJy6+lhfW2OUTqQeYSi5Ag0EYrVn/gEQALrFLQjCR3GjuHSindz0rd3Fnx/t7Sen
|
oWPzfBnkmiAN8KlZxHm5Ag0EYrVn/gEQALrFLQjCR3GjuHSindz0rd3Fnx/t7Sen
|
||||||
T+p07yCSSoSlmnJHCQmwh4vfg1blyz0zZ4vkIhtpHsEgc+ZAG+WQXSsJ2iRz+eSN
|
T+p07yCSSoSlmnJHCQmwh4vfg1blyz0zZ4vkIhtpHsEgc+ZAG+WQXSsJ2iRz+eSN
|
||||||
GwoOQl4XC3n+QWkc1ws+btr48+6UqXIQU+F8TPQyx/PIgi2nZXJB7f5+mjCqsk46
|
GwoOQl4XC3n+QWkc1ws+btr48+6UqXIQU+F8TPQyx/PIgi2nZXJB7f5+mjCqsk46
|
||||||
XvH4nTr4kJjuqMSR/++wvre2qNQRa/q/dTsK0OaN/mJsdX6Oi+aGNaQJUhIG7F+E
|
XvH4nTr4kJjuqMSR/++wvre2qNQRa/q/dTsK0OaN/mJsdX6Oi+aGNaQJUhIG7F+E
|
||||||
@ -64,20 +64,20 @@ qoExANj5lUTZPe8M50lI182FrcjAN7dClO3QI6pg7wy0erMxfFly3j8UQ91ysS9T
|
|||||||
s+GsP9I3cmWWQcKYxWHtE8xTXnNCVPFZQj2nwhJzae8ypfOtulBRA3dUKWGKuDH/
|
s+GsP9I3cmWWQcKYxWHtE8xTXnNCVPFZQj2nwhJzae8ypfOtulBRA3dUKWGKuDH/
|
||||||
axFENhUsT397aOU3qkP/od4a64JyNIEo4CTTSPVeWd7njsGqli2U3A4xL2CcyYvt
|
axFENhUsT397aOU3qkP/od4a64JyNIEo4CTTSPVeWd7njsGqli2U3A4xL2CcyYvt
|
||||||
D/MWcMBGEoLSNTswwKdom4FaJpn5KThnK/T0bQcmJblJhoCtppXisbexZnCpuS0x
|
D/MWcMBGEoLSNTswwKdom4FaJpn5KThnK/T0bQcmJblJhoCtppXisbexZnCpuS0x
|
||||||
Zdlm2T14KJ3LABEBAAGJAjwEGAEIACYWIQQ94D0eFEp/BpNZmdyq/SOBb80tWwUC
|
Zdlm2T14KJ3LABEBAAGJAjwEGAEIACYCGwwWIQQ94D0eFEp/BpNZmdyq/SOBb80t
|
||||||
YrVn/gIbDAUJA8JnAAAKCRCq/SOBb80tWyTBD/9AGpW6QoDF7zYjHAozH9S5RGCA
|
WwUCZoyyjQUJBbh+DwAKCRCq/SOBb80tW18XD/9MXztmf01MT+1kZdBouZ/7Rp/7
|
||||||
Y7E82dG/0xmFUwPprAG0BKmmgU6TiipyVGmKIXGYYYU92pMnbvXkYQMoa+WJNncN
|
9kuqo//B1G+RXau4oFtPqb67kNe2WaIc3u5B73PUHsMf3i6z4ib2KbMhZZerLn0O
|
||||||
D3fY52UeXeffTf4cFpStlzi9xgYtOLhFamzYu/4xhkjOX+xhOSXscCiFRyT8cF3B
|
dRglcuPeNWmsASY3dH/XVG0cT0zvvWegagd12TJEl3Vs+7XNrOw4cwDj9L1+GH9m
|
||||||
O6c5BHU+Zj0/rGPgOyPUbx7l7B9MubB/41nNX35k08e+8T3wtWDb4XF+15HnRfva
|
kSt4uaANWn/6a3RvMRhiVEYuNwhAzcKaactPmYqrLJgoVLbRSDkgyHaMQ2jKgLxk
|
||||||
6fblO8wgU25Orv2Rm1jnKGa9DxJ8nE40IMrqDapENtDuL+zKJsvR0+ptWvEyL56U
|
ifS/fvluGV0ub2Po6DJiqfRpd1tDvPhe9y1+r1WFDZsOcvTcZUfSt/7dXMGfqGu0
|
||||||
GtJJG5un6mXiLKuRQT0DEv4MdZRHDgDstDnqcbEiazVEbUuvhZZob6lRY2A19m1+
|
2daVFlfeSXSALrDE5uc0UxodHCpP3sqRYDZevGLBRaaTkIjYXG/+N898+7K5WJF4
|
||||||
7zfnDxkhqCA1RCnv4fdvcPdCMMFHwLpdhjgW0aI/uwgwrvsEz5+JRlnLvdQHlPAg
|
xXOLWxM2cwGkG7eC9pugcDnBp9XlF7O+GBiZ05JUe5flXDQFZ+h3exjopu6KHF1B
|
||||||
q7l2fGcBSpz9U0ayyfRPjPntsNCtZl1UDxGLeciPkZhyG84zEWQbk/j52ZpRN+Ik
|
RnzNy8LC0UKb+AuvRIOLV92a9Q9wGWU/jaVDu6nZ0umAeuSzxiHoDsonm0Fl9QAz
|
||||||
ALpRLa8RBFmFSmXDUmwQrmm1EmARyQXwweKU31hf8ZGbCp2lPuRYm1LuGiirXSVP
|
2/xCokebuoeLrEK7R2af3X86mqq3sVO4ax+HPYChzOaVQBiHUW/TAldWcldYYphR
|
||||||
GysjRAJgW+VRpBKOzFQoUAUbReVWSaCwT8s17THzf71DdDb6CTj31jMLLYWwBpA/
|
/e2WsbmQfvCRtz/bZfo+aUVnrHNjzVMtF2SszdVmA/04Y8pS28MqtuRqhm5DPOOd
|
||||||
i73DgobDZMIGEZZC1EKqza8eh11xfyHFzGec03tbh+lIen+5IiRtWiEWkDS9ll0G
|
g1YeUywK5jRZ1twyo1kzJEFPLaoeaXaycsR1PMVBW0Urik5mrR/pOWq7PPoZoKb2
|
||||||
zgS/ZdziCvdAutqnGA==
|
lXYLE8bwkuQTmsyL1g==
|
||||||
=gZWO
|
=9i7d
|
||||||
-----END PGP PUBLIC KEY BLOCK-----
|
-----END PGP PUBLIC KEY BLOCK-----
|
||||||
|
|
||||||
```
|
```
|
||||||
|
101
assets/go-licenses.json
generated
101
assets/go-licenses.json
generated
File diff suppressed because one or more lines are too long
@ -69,6 +69,7 @@ func newFileCollector(fileFilter string, batchSize int) (*fileCollector, error)
|
|||||||
co.includePatterns = append(co.includePatterns, regexp.MustCompile(`.*\.go$`))
|
co.includePatterns = append(co.includePatterns, regexp.MustCompile(`.*\.go$`))
|
||||||
|
|
||||||
co.excludePatterns = append(co.excludePatterns, regexp.MustCompile(`.*\bbindata\.go$`))
|
co.excludePatterns = append(co.excludePatterns, regexp.MustCompile(`.*\bbindata\.go$`))
|
||||||
|
co.excludePatterns = append(co.excludePatterns, regexp.MustCompile(`\.pb\.go$`))
|
||||||
co.excludePatterns = append(co.excludePatterns, regexp.MustCompile(`tests/gitea-repositories-meta`))
|
co.excludePatterns = append(co.excludePatterns, regexp.MustCompile(`tests/gitea-repositories-meta`))
|
||||||
co.excludePatterns = append(co.excludePatterns, regexp.MustCompile(`tests/integration/migration-test`))
|
co.excludePatterns = append(co.excludePatterns, regexp.MustCompile(`tests/integration/migration-test`))
|
||||||
co.excludePatterns = append(co.excludePatterns, regexp.MustCompile(`modules/git/tests`))
|
co.excludePatterns = append(co.excludePatterns, regexp.MustCompile(`modules/git/tests`))
|
||||||
@ -203,17 +204,6 @@ Example:
|
|||||||
`, "file-batch-exec")
|
`, "file-batch-exec")
|
||||||
}
|
}
|
||||||
|
|
||||||
func getGoVersion() string {
|
|
||||||
goModFile, err := os.ReadFile("go.mod")
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf(`Faild to read "go.mod": %v`, err)
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
goModVersionRegex := regexp.MustCompile(`go \d+\.\d+`)
|
|
||||||
goModVersionLine := goModVersionRegex.Find(goModFile)
|
|
||||||
return string(goModVersionLine[3:])
|
|
||||||
}
|
|
||||||
|
|
||||||
func newFileCollectorFromMainOptions(mainOptions map[string]string) (fc *fileCollector, err error) {
|
func newFileCollectorFromMainOptions(mainOptions map[string]string) (fc *fileCollector, err error) {
|
||||||
fileFilter := mainOptions["file-filter"]
|
fileFilter := mainOptions["file-filter"]
|
||||||
if fileFilter == "" {
|
if fileFilter == "" {
|
||||||
@ -278,7 +268,8 @@ func main() {
|
|||||||
log.Print("the -d option is not supported by gitea-fmt")
|
log.Print("the -d option is not supported by gitea-fmt")
|
||||||
}
|
}
|
||||||
cmdErrors = append(cmdErrors, giteaFormatGoImports(files, containsString(subArgs, "-w")))
|
cmdErrors = append(cmdErrors, giteaFormatGoImports(files, containsString(subArgs, "-w")))
|
||||||
cmdErrors = append(cmdErrors, passThroughCmd("go", append([]string{"run", os.Getenv("GOFUMPT_PACKAGE"), "-extra", "-lang", getGoVersion()}, substArgs...)))
|
cmdErrors = append(cmdErrors, passThroughCmd("gofmt", append([]string{"-w", "-r", "interface{} -> any"}, substArgs...)))
|
||||||
|
cmdErrors = append(cmdErrors, passThroughCmd("go", append([]string{"run", os.Getenv("GOFUMPT_PACKAGE"), "-extra"}, substArgs...)))
|
||||||
default:
|
default:
|
||||||
log.Fatalf("unknown cmd: %s %v", subCmd, subArgs)
|
log.Fatalf("unknown cmd: %s %v", subCmd, subArgs)
|
||||||
}
|
}
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
package cmd
|
package cmd
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"text/tabwriter"
|
"text/tabwriter"
|
||||||
@ -91,7 +92,7 @@ func runListAuth(c *cli.Context) error {
|
|||||||
|
|
||||||
func runDeleteAuth(c *cli.Context) error {
|
func runDeleteAuth(c *cli.Context) error {
|
||||||
if !c.IsSet("id") {
|
if !c.IsSet("id") {
|
||||||
return fmt.Errorf("--id flag is missing")
|
return errors.New("--id flag is missing")
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx, cancel := installSignals()
|
ctx, cancel := installSignals()
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
package cmd
|
package cmd
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
|
||||||
@ -193,7 +194,7 @@ func runAddOauth(c *cli.Context) error {
|
|||||||
|
|
||||||
func runUpdateOauth(c *cli.Context) error {
|
func runUpdateOauth(c *cli.Context) error {
|
||||||
if !c.IsSet("id") {
|
if !c.IsSet("id") {
|
||||||
return fmt.Errorf("--id flag is missing")
|
return errors.New("--id flag is missing")
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx, cancel := installSignals()
|
ctx, cancel := installSignals()
|
||||||
|
@ -5,7 +5,6 @@ package cmd
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
auth_model "code.gitea.io/gitea/models/auth"
|
auth_model "code.gitea.io/gitea/models/auth"
|
||||||
@ -166,7 +165,7 @@ func runAddSMTP(c *cli.Context) error {
|
|||||||
|
|
||||||
func runUpdateSMTP(c *cli.Context) error {
|
func runUpdateSMTP(c *cli.Context) error {
|
||||||
if !c.IsSet("id") {
|
if !c.IsSet("id") {
|
||||||
return fmt.Errorf("--id flag is missing")
|
return errors.New("--id flag is missing")
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx, cancel := installSignals()
|
ctx, cancel := installSignals()
|
||||||
|
@ -35,7 +35,8 @@ var microcmdUserChangePassword = &cli.Command{
|
|||||||
},
|
},
|
||||||
&cli.BoolFlag{
|
&cli.BoolFlag{
|
||||||
Name: "must-change-password",
|
Name: "must-change-password",
|
||||||
Usage: "User must change password",
|
Usage: "User must change password (can be disabled by --must-change-password=false)",
|
||||||
|
Value: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@ -57,23 +58,18 @@ func runChangePassword(c *cli.Context) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
var mustChangePassword optional.Option[bool]
|
|
||||||
if c.IsSet("must-change-password") {
|
|
||||||
mustChangePassword = optional.Some(c.Bool("must-change-password"))
|
|
||||||
}
|
|
||||||
|
|
||||||
opts := &user_service.UpdateAuthOptions{
|
opts := &user_service.UpdateAuthOptions{
|
||||||
Password: optional.Some(c.String("password")),
|
Password: optional.Some(c.String("password")),
|
||||||
MustChangePassword: mustChangePassword,
|
MustChangePassword: optional.Some(c.Bool("must-change-password")),
|
||||||
}
|
}
|
||||||
if err := user_service.UpdateAuth(ctx, user, opts); err != nil {
|
if err := user_service.UpdateAuth(ctx, user, opts); err != nil {
|
||||||
switch {
|
switch {
|
||||||
case errors.Is(err, password.ErrMinLength):
|
case errors.Is(err, password.ErrMinLength):
|
||||||
return fmt.Errorf("Password is not long enough. Needs to be at least %d", setting.MinPasswordLength)
|
return fmt.Errorf("password is not long enough, needs to be at least %d characters", setting.MinPasswordLength)
|
||||||
case errors.Is(err, password.ErrComplexity):
|
case errors.Is(err, password.ErrComplexity):
|
||||||
return errors.New("Password does not meet complexity requirements")
|
return errors.New("password does not meet complexity requirements")
|
||||||
case errors.Is(err, password.ErrIsPwned):
|
case errors.Is(err, password.ErrIsPwned):
|
||||||
return errors.New("The password you chose is on a list of stolen passwords previously exposed in public data breaches. Please try again with a different password.\nFor more details, see https://haveibeenpwned.com/Passwords")
|
return errors.New("the password is in a list of stolen passwords previously exposed in public data breaches, please try again with a different password, to see more details: https://haveibeenpwned.com/Passwords")
|
||||||
default:
|
default:
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -4,10 +4,12 @@
|
|||||||
package cmd
|
package cmd
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
auth_model "code.gitea.io/gitea/models/auth"
|
auth_model "code.gitea.io/gitea/models/auth"
|
||||||
|
"code.gitea.io/gitea/models/db"
|
||||||
user_model "code.gitea.io/gitea/models/user"
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
pwd "code.gitea.io/gitea/modules/auth/password"
|
pwd "code.gitea.io/gitea/modules/auth/password"
|
||||||
"code.gitea.io/gitea/modules/optional"
|
"code.gitea.io/gitea/modules/optional"
|
||||||
@ -46,8 +48,9 @@ var microcmdUserCreate = &cli.Command{
|
|||||||
Usage: "Generate a random password for the user",
|
Usage: "Generate a random password for the user",
|
||||||
},
|
},
|
||||||
&cli.BoolFlag{
|
&cli.BoolFlag{
|
||||||
Name: "must-change-password",
|
Name: "must-change-password",
|
||||||
Usage: "Set this option to false to prevent forcing the user to change their password after initial login, (Default: true)",
|
Usage: "User must change password after initial login, defaults to true for all users except the first one (can be disabled by --must-change-password=false)",
|
||||||
|
DisableDefaultText: true,
|
||||||
},
|
},
|
||||||
&cli.IntFlag{
|
&cli.IntFlag{
|
||||||
Name: "random-password-length",
|
Name: "random-password-length",
|
||||||
@ -71,10 +74,10 @@ func runCreateUser(c *cli.Context) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if c.IsSet("name") && c.IsSet("username") {
|
if c.IsSet("name") && c.IsSet("username") {
|
||||||
return errors.New("Cannot set both --name and --username flags")
|
return errors.New("cannot set both --name and --username flags")
|
||||||
}
|
}
|
||||||
if !c.IsSet("name") && !c.IsSet("username") {
|
if !c.IsSet("name") && !c.IsSet("username") {
|
||||||
return errors.New("One of --name or --username flags must be set")
|
return errors.New("one of --name or --username flags must be set")
|
||||||
}
|
}
|
||||||
|
|
||||||
if c.IsSet("password") && c.IsSet("random-password") {
|
if c.IsSet("password") && c.IsSet("random-password") {
|
||||||
@ -89,11 +92,16 @@ func runCreateUser(c *cli.Context) error {
|
|||||||
_, _ = fmt.Fprintf(c.App.ErrWriter, "--name flag is deprecated. Use --username instead.\n")
|
_, _ = fmt.Fprintf(c.App.ErrWriter, "--name flag is deprecated. Use --username instead.\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx, cancel := installSignals()
|
ctx := c.Context
|
||||||
defer cancel()
|
if !setting.IsInTesting {
|
||||||
|
// FIXME: need to refactor the "installSignals/initDB" related code later
|
||||||
if err := initDB(ctx); err != nil {
|
// it doesn't make sense to call it in (almost) every command action function
|
||||||
return err
|
var cancel context.CancelFunc
|
||||||
|
ctx, cancel = installSignals()
|
||||||
|
defer cancel()
|
||||||
|
if err := initDB(ctx); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var password string
|
var password string
|
||||||
@ -110,17 +118,21 @@ func runCreateUser(c *cli.Context) error {
|
|||||||
return errors.New("must set either password or random-password flag")
|
return errors.New("must set either password or random-password flag")
|
||||||
}
|
}
|
||||||
|
|
||||||
// always default to true
|
isAdmin := c.Bool("admin")
|
||||||
changePassword := true
|
mustChangePassword := true // always default to true
|
||||||
|
|
||||||
// If this is the first user being created.
|
|
||||||
// Take it as the admin and don't force a password update.
|
|
||||||
if n := user_model.CountUsers(ctx, nil); n == 0 {
|
|
||||||
changePassword = false
|
|
||||||
}
|
|
||||||
|
|
||||||
if c.IsSet("must-change-password") {
|
if c.IsSet("must-change-password") {
|
||||||
changePassword = c.Bool("must-change-password")
|
// if the flag is set, use the value provided by the user
|
||||||
|
mustChangePassword = c.Bool("must-change-password")
|
||||||
|
} else {
|
||||||
|
// check whether there are users in the database
|
||||||
|
hasUserRecord, err := db.IsTableNotEmpty(&user_model.User{})
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("IsTableNotEmpty: %w", err)
|
||||||
|
}
|
||||||
|
if !hasUserRecord {
|
||||||
|
// if this is the first one being created, don't force to change password (keep the old behavior)
|
||||||
|
mustChangePassword = false
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
restricted := optional.None[bool]()
|
restricted := optional.None[bool]()
|
||||||
@ -136,8 +148,8 @@ func runCreateUser(c *cli.Context) error {
|
|||||||
Name: username,
|
Name: username,
|
||||||
Email: c.String("email"),
|
Email: c.String("email"),
|
||||||
Passwd: password,
|
Passwd: password,
|
||||||
IsAdmin: c.Bool("admin"),
|
IsAdmin: isAdmin,
|
||||||
MustChangePassword: changePassword,
|
MustChangePassword: mustChangePassword,
|
||||||
Visibility: visibility,
|
Visibility: visibility,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
44
cmd/admin_user_create_test.go
Normal file
44
cmd/admin_user_create_test.go
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
// Copyright 2023 The Gitea Authors. All rights reserved.
|
||||||
|
// SPDX-License-Identifier: MIT
|
||||||
|
|
||||||
|
package cmd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"code.gitea.io/gitea/models/db"
|
||||||
|
"code.gitea.io/gitea/models/unittest"
|
||||||
|
user_model "code.gitea.io/gitea/models/user"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestAdminUserCreate(t *testing.T) {
|
||||||
|
app := NewMainApp(AppVersion{})
|
||||||
|
|
||||||
|
reset := func() {
|
||||||
|
assert.NoError(t, db.TruncateBeans(db.DefaultContext, &user_model.User{}))
|
||||||
|
assert.NoError(t, db.TruncateBeans(db.DefaultContext, &user_model.EmailAddress{}))
|
||||||
|
}
|
||||||
|
|
||||||
|
type createCheck struct{ IsAdmin, MustChangePassword bool }
|
||||||
|
createUser := func(name, args string) createCheck {
|
||||||
|
assert.NoError(t, app.Run(strings.Fields(fmt.Sprintf("./gitea admin user create --username %s --email %s@gitea.local %s --password foobar", name, name, args))))
|
||||||
|
u := unittest.AssertExistsAndLoadBean(t, &user_model.User{LowerName: name})
|
||||||
|
return createCheck{u.IsAdmin, u.MustChangePassword}
|
||||||
|
}
|
||||||
|
reset()
|
||||||
|
assert.Equal(t, createCheck{IsAdmin: false, MustChangePassword: false}, createUser("u", ""), "first non-admin user doesn't need to change password")
|
||||||
|
|
||||||
|
reset()
|
||||||
|
assert.Equal(t, createCheck{IsAdmin: true, MustChangePassword: false}, createUser("u", "--admin"), "first admin user doesn't need to change password")
|
||||||
|
|
||||||
|
reset()
|
||||||
|
assert.Equal(t, createCheck{IsAdmin: true, MustChangePassword: true}, createUser("u", "--admin --must-change-password"))
|
||||||
|
assert.Equal(t, createCheck{IsAdmin: true, MustChangePassword: true}, createUser("u2", "--admin"))
|
||||||
|
assert.Equal(t, createCheck{IsAdmin: true, MustChangePassword: false}, createUser("u3", "--admin --must-change-password=false"))
|
||||||
|
assert.Equal(t, createCheck{IsAdmin: false, MustChangePassword: true}, createUser("u4", ""))
|
||||||
|
assert.Equal(t, createCheck{IsAdmin: false, MustChangePassword: false}, createUser("u5", "--must-change-password=false"))
|
||||||
|
}
|
@ -4,6 +4,7 @@
|
|||||||
package cmd
|
package cmd
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
@ -42,7 +43,7 @@ var microcmdUserDelete = &cli.Command{
|
|||||||
|
|
||||||
func runDeleteUser(c *cli.Context) error {
|
func runDeleteUser(c *cli.Context) error {
|
||||||
if !c.IsSet("id") && !c.IsSet("username") && !c.IsSet("email") {
|
if !c.IsSet("id") && !c.IsSet("username") && !c.IsSet("email") {
|
||||||
return fmt.Errorf("You must provide the id, username or email of a user to delete")
|
return errors.New("You must provide the id, username or email of a user to delete")
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx, cancel := installSignals()
|
ctx, cancel := installSignals()
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
package cmd
|
package cmd
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
auth_model "code.gitea.io/gitea/models/auth"
|
auth_model "code.gitea.io/gitea/models/auth"
|
||||||
@ -42,7 +43,7 @@ var microcmdUserGenerateAccessToken = &cli.Command{
|
|||||||
|
|
||||||
func runGenerateAccessToken(c *cli.Context) error {
|
func runGenerateAccessToken(c *cli.Context) error {
|
||||||
if !c.IsSet("username") {
|
if !c.IsSet("username") {
|
||||||
return fmt.Errorf("You must provide a username to generate a token for")
|
return errors.New("You must provide a username to generate a token for")
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx, cancel := installSignals()
|
ctx, cancel := installSignals()
|
||||||
@ -68,7 +69,7 @@ func runGenerateAccessToken(c *cli.Context) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
if exist {
|
if exist {
|
||||||
return fmt.Errorf("access token name has been used already")
|
return errors.New("access token name has been used already")
|
||||||
}
|
}
|
||||||
|
|
||||||
// make sure the scopes are valid
|
// make sure the scopes are valid
|
||||||
|
362
cmd/dump.go
362
cmd/dump.go
@ -6,14 +6,13 @@ package cmd
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
|
||||||
|
|
||||||
"code.gitea.io/gitea/models/db"
|
"code.gitea.io/gitea/models/db"
|
||||||
|
"code.gitea.io/gitea/modules/dump"
|
||||||
"code.gitea.io/gitea/modules/json"
|
"code.gitea.io/gitea/modules/json"
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
@ -25,89 +24,17 @@ import (
|
|||||||
"github.com/urfave/cli/v2"
|
"github.com/urfave/cli/v2"
|
||||||
)
|
)
|
||||||
|
|
||||||
func addReader(w archiver.Writer, r io.ReadCloser, info os.FileInfo, customName string, verbose bool) error {
|
|
||||||
if verbose {
|
|
||||||
log.Info("Adding file %s", customName)
|
|
||||||
}
|
|
||||||
|
|
||||||
return w.Write(archiver.File{
|
|
||||||
FileInfo: archiver.FileInfo{
|
|
||||||
FileInfo: info,
|
|
||||||
CustomName: customName,
|
|
||||||
},
|
|
||||||
ReadCloser: r,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func addFile(w archiver.Writer, filePath, absPath string, verbose bool) error {
|
|
||||||
file, err := os.Open(absPath)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer file.Close()
|
|
||||||
fileInfo, err := file.Stat()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return addReader(w, file, fileInfo, filePath, verbose)
|
|
||||||
}
|
|
||||||
|
|
||||||
func isSubdir(upper, lower string) (bool, error) {
|
|
||||||
if relPath, err := filepath.Rel(upper, lower); err != nil {
|
|
||||||
return false, err
|
|
||||||
} else if relPath == "." || !strings.HasPrefix(relPath, ".") {
|
|
||||||
return true, nil
|
|
||||||
}
|
|
||||||
return false, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
type outputType struct {
|
|
||||||
Enum []string
|
|
||||||
Default string
|
|
||||||
selected string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (o outputType) Join() string {
|
|
||||||
return strings.Join(o.Enum, ", ")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (o *outputType) Set(value string) error {
|
|
||||||
for _, enum := range o.Enum {
|
|
||||||
if enum == value {
|
|
||||||
o.selected = value
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return fmt.Errorf("allowed values are %s", o.Join())
|
|
||||||
}
|
|
||||||
|
|
||||||
func (o outputType) String() string {
|
|
||||||
if o.selected == "" {
|
|
||||||
return o.Default
|
|
||||||
}
|
|
||||||
return o.selected
|
|
||||||
}
|
|
||||||
|
|
||||||
var outputTypeEnum = &outputType{
|
|
||||||
Enum: []string{"zip", "tar", "tar.sz", "tar.gz", "tar.xz", "tar.bz2", "tar.br", "tar.lz4", "tar.zst"},
|
|
||||||
Default: "zip",
|
|
||||||
}
|
|
||||||
|
|
||||||
// CmdDump represents the available dump sub-command.
|
// CmdDump represents the available dump sub-command.
|
||||||
var CmdDump = &cli.Command{
|
var CmdDump = &cli.Command{
|
||||||
Name: "dump",
|
Name: "dump",
|
||||||
Usage: "Dump Gitea files and database",
|
Usage: "Dump Gitea files and database",
|
||||||
Description: `Dump compresses all related files and database into zip file.
|
Description: `Dump compresses all related files and database into zip file. It can be used for backup and capture Gitea server image to send to maintainer`,
|
||||||
It can be used for backup and capture Gitea server image to send to maintainer`,
|
Action: runDump,
|
||||||
Action: runDump,
|
|
||||||
Flags: []cli.Flag{
|
Flags: []cli.Flag{
|
||||||
&cli.StringFlag{
|
&cli.StringFlag{
|
||||||
Name: "file",
|
Name: "file",
|
||||||
Aliases: []string{"f"},
|
Aliases: []string{"f"},
|
||||||
Value: fmt.Sprintf("gitea-dump-%d.zip", time.Now().Unix()),
|
Usage: `Name of the dump file which will be created, default to "gitea-dump-{time}.zip". Supply '-' for stdout. See type for available types.`,
|
||||||
Usage: "Name of the dump file which will be created. Supply '-' for stdout. See type for available types.",
|
|
||||||
},
|
},
|
||||||
&cli.BoolFlag{
|
&cli.BoolFlag{
|
||||||
Name: "verbose",
|
Name: "verbose",
|
||||||
@ -160,64 +87,56 @@ It can be used for backup and capture Gitea server image to send to maintainer`,
|
|||||||
Name: "skip-index",
|
Name: "skip-index",
|
||||||
Usage: "Skip bleve index data",
|
Usage: "Skip bleve index data",
|
||||||
},
|
},
|
||||||
&cli.GenericFlag{
|
&cli.BoolFlag{
|
||||||
|
Name: "skip-db",
|
||||||
|
Usage: "Skip database",
|
||||||
|
},
|
||||||
|
&cli.StringFlag{
|
||||||
Name: "type",
|
Name: "type",
|
||||||
Value: outputTypeEnum,
|
Usage: fmt.Sprintf(`Dump output format, default to "zip", supported types: %s`, strings.Join(dump.SupportedOutputTypes, ", ")),
|
||||||
Usage: fmt.Sprintf("Dump output format: %s", outputTypeEnum.Join()),
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
func fatal(format string, args ...any) {
|
func fatal(format string, args ...any) {
|
||||||
fmt.Fprintf(os.Stderr, format+"\n", args...)
|
|
||||||
log.Fatal(format, args...)
|
log.Fatal(format, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func runDump(ctx *cli.Context) error {
|
func runDump(ctx *cli.Context) error {
|
||||||
var file *os.File
|
|
||||||
fileName := ctx.String("file")
|
|
||||||
outType := ctx.String("type")
|
|
||||||
if fileName == "-" {
|
|
||||||
file = os.Stdout
|
|
||||||
setupConsoleLogger(log.FATAL, log.CanColorStderr, os.Stderr)
|
|
||||||
} else {
|
|
||||||
for _, suffix := range outputTypeEnum.Enum {
|
|
||||||
if strings.HasSuffix(fileName, "."+suffix) {
|
|
||||||
fileName = strings.TrimSuffix(fileName, "."+suffix)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fileName += "." + outType
|
|
||||||
}
|
|
||||||
setting.MustInstalled()
|
setting.MustInstalled()
|
||||||
|
|
||||||
// make sure we are logging to the console no matter what the configuration tells us do to
|
quite := ctx.Bool("quiet")
|
||||||
// FIXME: don't use CfgProvider directly
|
|
||||||
if _, err := setting.CfgProvider.Section("log").NewKey("MODE", "console"); err != nil {
|
|
||||||
fatal("Setting logging mode to console failed: %v", err)
|
|
||||||
}
|
|
||||||
if _, err := setting.CfgProvider.Section("log.console").NewKey("STDERR", "true"); err != nil {
|
|
||||||
fatal("Setting console logger to stderr failed: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set loglevel to Warn if quiet-mode is requested
|
|
||||||
if ctx.Bool("quiet") {
|
|
||||||
if _, err := setting.CfgProvider.Section("log.console").NewKey("LEVEL", "Warn"); err != nil {
|
|
||||||
fatal("Setting console log-level failed: %v", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if !setting.InstallLock {
|
|
||||||
log.Error("Is '%s' really the right config path?\n", setting.CustomConf)
|
|
||||||
return fmt.Errorf("gitea is not initialized")
|
|
||||||
}
|
|
||||||
setting.LoadSettings() // cannot access session settings otherwise
|
|
||||||
|
|
||||||
verbose := ctx.Bool("verbose")
|
verbose := ctx.Bool("verbose")
|
||||||
if verbose && ctx.Bool("quiet") {
|
if verbose && quite {
|
||||||
return fmt.Errorf("--quiet and --verbose cannot both be set")
|
fatal("Option --quiet and --verbose cannot both be set")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// outFileName is either "-" or a file name (will be made absolute)
|
||||||
|
outFileName, outType := dump.PrepareFileNameAndType(ctx.String("file"), ctx.String("type"))
|
||||||
|
if outType == "" {
|
||||||
|
fatal("Invalid output type")
|
||||||
|
}
|
||||||
|
|
||||||
|
outFile := os.Stdout
|
||||||
|
if outFileName != "-" {
|
||||||
|
var err error
|
||||||
|
if outFileName, err = filepath.Abs(outFileName); err != nil {
|
||||||
|
fatal("Unable to get absolute path of dump file: %v", err)
|
||||||
|
}
|
||||||
|
if exist, _ := util.IsExist(outFileName); exist {
|
||||||
|
fatal("Dump file %q exists", outFileName)
|
||||||
|
}
|
||||||
|
if outFile, err = os.Create(outFileName); err != nil {
|
||||||
|
fatal("Unable to create dump file %q: %v", outFileName, err)
|
||||||
|
}
|
||||||
|
defer outFile.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
setupConsoleLogger(util.Iif(quite, log.WARN, log.INFO), log.CanColorStderr, os.Stderr)
|
||||||
|
|
||||||
|
setting.DisableLoggerInit()
|
||||||
|
setting.LoadSettings() // cannot access session settings otherwise
|
||||||
|
|
||||||
stdCtx, cancel := installSignals()
|
stdCtx, cancel := installSignals()
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
@ -226,44 +145,32 @@ func runDump(ctx *cli.Context) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := storage.Init(); err != nil {
|
if err = storage.Init(); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if file == nil {
|
archiverGeneric, err := archiver.ByExtension("." + outType)
|
||||||
file, err = os.Create(fileName)
|
|
||||||
if err != nil {
|
|
||||||
fatal("Unable to open %s: %v", fileName, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
defer file.Close()
|
|
||||||
|
|
||||||
absFileName, err := filepath.Abs(fileName)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
var iface any
|
|
||||||
if fileName == "-" {
|
|
||||||
iface, err = archiver.ByExtension(fmt.Sprintf(".%s", outType))
|
|
||||||
} else {
|
|
||||||
iface, err = archiver.ByExtension(fileName)
|
|
||||||
}
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fatal("Unable to get archiver for extension: %v", err)
|
fatal("Unable to get archiver for extension: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
w, _ := iface.(archiver.Writer)
|
archiverWriter := archiverGeneric.(archiver.Writer)
|
||||||
if err := w.Create(file); err != nil {
|
if err := archiverWriter.Create(outFile); err != nil {
|
||||||
fatal("Creating archiver.Writer failed: %v", err)
|
fatal("Creating archiver.Writer failed: %v", err)
|
||||||
}
|
}
|
||||||
defer w.Close()
|
defer archiverWriter.Close()
|
||||||
|
|
||||||
|
dumper := &dump.Dumper{
|
||||||
|
Writer: archiverWriter,
|
||||||
|
Verbose: verbose,
|
||||||
|
}
|
||||||
|
dumper.GlobalExcludeAbsPath(outFileName)
|
||||||
|
|
||||||
if ctx.IsSet("skip-repository") && ctx.Bool("skip-repository") {
|
if ctx.IsSet("skip-repository") && ctx.Bool("skip-repository") {
|
||||||
log.Info("Skip dumping local repositories")
|
log.Info("Skip dumping local repositories")
|
||||||
} else {
|
} else {
|
||||||
log.Info("Dumping local repositories... %s", setting.RepoRootPath)
|
log.Info("Dumping local repositories... %s", setting.RepoRootPath)
|
||||||
if err := addRecursiveExclude(w, "repos", setting.RepoRootPath, []string{absFileName}, verbose); err != nil {
|
if err := dumper.AddRecursiveExclude("repos", setting.RepoRootPath, nil); err != nil {
|
||||||
fatal("Failed to include repositories: %v", err)
|
fatal("Failed to include repositories: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -276,49 +183,52 @@ func runDump(ctx *cli.Context) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
return dumper.AddReader(object, info, path.Join("data", "lfs", objPath))
|
||||||
return addReader(w, object, info, path.Join("data", "lfs", objPath), verbose)
|
|
||||||
}); err != nil {
|
}); err != nil {
|
||||||
fatal("Failed to dump LFS objects: %v", err)
|
fatal("Failed to dump LFS objects: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
tmpDir := ctx.String("tempdir")
|
if ctx.Bool("skip-db") {
|
||||||
if _, err := os.Stat(tmpDir); os.IsNotExist(err) {
|
// Ensure that we don't dump the database file that may reside in setting.AppDataPath or elsewhere.
|
||||||
fatal("Path does not exist: %s", tmpDir)
|
dumper.GlobalExcludeAbsPath(setting.Database.Path)
|
||||||
}
|
log.Info("Skipping database")
|
||||||
|
|
||||||
dbDump, err := os.CreateTemp(tmpDir, "gitea-db.sql")
|
|
||||||
if err != nil {
|
|
||||||
fatal("Failed to create tmp file: %v", err)
|
|
||||||
}
|
|
||||||
defer func() {
|
|
||||||
_ = dbDump.Close()
|
|
||||||
if err := util.Remove(dbDump.Name()); err != nil {
|
|
||||||
log.Warn("Unable to remove temporary file: %s: Error: %v", dbDump.Name(), err)
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
targetDBType := ctx.String("database")
|
|
||||||
if len(targetDBType) > 0 && targetDBType != setting.Database.Type.String() {
|
|
||||||
log.Info("Dumping database %s => %s...", setting.Database.Type, targetDBType)
|
|
||||||
} else {
|
} else {
|
||||||
log.Info("Dumping database...")
|
tmpDir := ctx.String("tempdir")
|
||||||
}
|
if _, err := os.Stat(tmpDir); os.IsNotExist(err) {
|
||||||
|
fatal("Path does not exist: %s", tmpDir)
|
||||||
if err := db.DumpDatabase(dbDump.Name(), targetDBType); err != nil {
|
|
||||||
fatal("Failed to dump database: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := addFile(w, "gitea-db.sql", dbDump.Name(), verbose); err != nil {
|
|
||||||
fatal("Failed to include gitea-db.sql: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(setting.CustomConf) > 0 {
|
|
||||||
log.Info("Adding custom configuration file from %s", setting.CustomConf)
|
|
||||||
if err := addFile(w, "app.ini", setting.CustomConf, verbose); err != nil {
|
|
||||||
fatal("Failed to include specified app.ini: %v", err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
dbDump, err := os.CreateTemp(tmpDir, "gitea-db.sql")
|
||||||
|
if err != nil {
|
||||||
|
fatal("Failed to create tmp file: %v", err)
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
_ = dbDump.Close()
|
||||||
|
if err := util.Remove(dbDump.Name()); err != nil {
|
||||||
|
log.Warn("Unable to remove temporary file: %s: Error: %v", dbDump.Name(), err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
targetDBType := ctx.String("database")
|
||||||
|
if len(targetDBType) > 0 && targetDBType != setting.Database.Type.String() {
|
||||||
|
log.Info("Dumping database %s => %s...", setting.Database.Type, targetDBType)
|
||||||
|
} else {
|
||||||
|
log.Info("Dumping database...")
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := db.DumpDatabase(dbDump.Name(), targetDBType); err != nil {
|
||||||
|
fatal("Failed to dump database: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = dumper.AddFile("gitea-db.sql", dbDump.Name()); err != nil {
|
||||||
|
fatal("Failed to include gitea-db.sql: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Info("Adding custom configuration file from %s", setting.CustomConf)
|
||||||
|
if err = dumper.AddFile("app.ini", setting.CustomConf); err != nil {
|
||||||
|
fatal("Failed to include specified app.ini: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if ctx.IsSet("skip-custom-dir") && ctx.Bool("skip-custom-dir") {
|
if ctx.IsSet("skip-custom-dir") && ctx.Bool("skip-custom-dir") {
|
||||||
@ -326,8 +236,8 @@ func runDump(ctx *cli.Context) error {
|
|||||||
} else {
|
} else {
|
||||||
customDir, err := os.Stat(setting.CustomPath)
|
customDir, err := os.Stat(setting.CustomPath)
|
||||||
if err == nil && customDir.IsDir() {
|
if err == nil && customDir.IsDir() {
|
||||||
if is, _ := isSubdir(setting.AppDataPath, setting.CustomPath); !is {
|
if is, _ := dump.IsSubdir(setting.AppDataPath, setting.CustomPath); !is {
|
||||||
if err := addRecursiveExclude(w, "custom", setting.CustomPath, []string{absFileName}, verbose); err != nil {
|
if err := dumper.AddRecursiveExclude("custom", setting.CustomPath, nil); err != nil {
|
||||||
fatal("Failed to include custom: %v", err)
|
fatal("Failed to include custom: %v", err)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -364,8 +274,7 @@ func runDump(ctx *cli.Context) error {
|
|||||||
excludes = append(excludes, setting.Attachment.Storage.Path)
|
excludes = append(excludes, setting.Attachment.Storage.Path)
|
||||||
excludes = append(excludes, setting.Packages.Storage.Path)
|
excludes = append(excludes, setting.Packages.Storage.Path)
|
||||||
excludes = append(excludes, setting.Log.RootPath)
|
excludes = append(excludes, setting.Log.RootPath)
|
||||||
excludes = append(excludes, absFileName)
|
if err := dumper.AddRecursiveExclude("data", setting.AppDataPath, excludes); err != nil {
|
||||||
if err := addRecursiveExclude(w, "data", setting.AppDataPath, excludes, verbose); err != nil {
|
|
||||||
fatal("Failed to include data directory: %v", err)
|
fatal("Failed to include data directory: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -377,8 +286,7 @@ func runDump(ctx *cli.Context) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
return dumper.AddReader(object, info, path.Join("data", "attachments", objPath))
|
||||||
return addReader(w, object, info, path.Join("data", "attachments", objPath), verbose)
|
|
||||||
}); err != nil {
|
}); err != nil {
|
||||||
fatal("Failed to dump attachments: %v", err)
|
fatal("Failed to dump attachments: %v", err)
|
||||||
}
|
}
|
||||||
@ -392,8 +300,7 @@ func runDump(ctx *cli.Context) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
return dumper.AddReader(object, info, path.Join("data", "packages", objPath))
|
||||||
return addReader(w, object, info, path.Join("data", "packages", objPath), verbose)
|
|
||||||
}); err != nil {
|
}); err != nil {
|
||||||
fatal("Failed to dump packages: %v", err)
|
fatal("Failed to dump packages: %v", err)
|
||||||
}
|
}
|
||||||
@ -409,80 +316,23 @@ func runDump(ctx *cli.Context) error {
|
|||||||
log.Error("Unable to check if %s exists. Error: %v", setting.Log.RootPath, err)
|
log.Error("Unable to check if %s exists. Error: %v", setting.Log.RootPath, err)
|
||||||
}
|
}
|
||||||
if isExist {
|
if isExist {
|
||||||
if err := addRecursiveExclude(w, "log", setting.Log.RootPath, []string{absFileName}, verbose); err != nil {
|
if err := dumper.AddRecursiveExclude("log", setting.Log.RootPath, nil); err != nil {
|
||||||
fatal("Failed to include log: %v", err)
|
fatal("Failed to include log: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if fileName != "-" {
|
if outFileName == "-" {
|
||||||
if err = w.Close(); err != nil {
|
log.Info("Finish dumping to stdout")
|
||||||
_ = util.Remove(fileName)
|
} else {
|
||||||
fatal("Failed to save %s: %v", fileName, err)
|
if err = archiverWriter.Close(); err != nil {
|
||||||
|
_ = os.Remove(outFileName)
|
||||||
|
fatal("Failed to save %q: %v", outFileName, err)
|
||||||
}
|
}
|
||||||
|
if err = os.Chmod(outFileName, 0o600); err != nil {
|
||||||
if err := os.Chmod(fileName, 0o600); err != nil {
|
|
||||||
log.Info("Can't change file access permissions mask to 0600: %v", err)
|
log.Info("Can't change file access permissions mask to 0600: %v", err)
|
||||||
}
|
}
|
||||||
}
|
log.Info("Finish dumping in file %s", outFileName)
|
||||||
|
|
||||||
if fileName != "-" {
|
|
||||||
log.Info("Finish dumping in file %s", fileName)
|
|
||||||
} else {
|
|
||||||
log.Info("Finish dumping to stdout")
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// addRecursiveExclude zips absPath to specified insidePath inside writer excluding excludeAbsPath
|
|
||||||
func addRecursiveExclude(w archiver.Writer, insidePath, absPath string, excludeAbsPath []string, verbose bool) error {
|
|
||||||
absPath, err := filepath.Abs(absPath)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
dir, err := os.Open(absPath)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer dir.Close()
|
|
||||||
|
|
||||||
files, err := dir.Readdir(0)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
for _, file := range files {
|
|
||||||
currentAbsPath := filepath.Join(absPath, file.Name())
|
|
||||||
currentInsidePath := path.Join(insidePath, file.Name())
|
|
||||||
if file.IsDir() {
|
|
||||||
if !util.SliceContainsString(excludeAbsPath, currentAbsPath) {
|
|
||||||
if err := addFile(w, currentInsidePath, currentAbsPath, false); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err = addRecursiveExclude(w, currentInsidePath, currentAbsPath, excludeAbsPath, verbose); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// only copy regular files and symlink regular files, skip non-regular files like socket/pipe/...
|
|
||||||
shouldAdd := file.Mode().IsRegular()
|
|
||||||
if !shouldAdd && file.Mode()&os.ModeSymlink == os.ModeSymlink {
|
|
||||||
target, err := filepath.EvalSymlinks(currentAbsPath)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
targetStat, err := os.Stat(target)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
shouldAdd = targetStat.Mode().IsRegular()
|
|
||||||
}
|
|
||||||
if shouldAdd {
|
|
||||||
if err = addFile(w, currentInsidePath, currentAbsPath, verbose); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -157,9 +157,9 @@ func runViewDo(c *cli.Context) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if len(matchedAssetFiles) == 0 {
|
if len(matchedAssetFiles) == 0 {
|
||||||
return fmt.Errorf("no files matched the given pattern")
|
return errors.New("no files matched the given pattern")
|
||||||
} else if len(matchedAssetFiles) > 1 {
|
} else if len(matchedAssetFiles) > 1 {
|
||||||
return fmt.Errorf("too many files matched the given pattern, try to be more specific")
|
return errors.New("too many files matched the given pattern, try to be more specific")
|
||||||
}
|
}
|
||||||
|
|
||||||
data, err := matchedAssetFiles[0].fs.ReadFile(matchedAssetFiles[0].name)
|
data, err := matchedAssetFiles[0].fs.ReadFile(matchedAssetFiles[0].name)
|
||||||
@ -180,7 +180,7 @@ func runExtractDo(c *cli.Context) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if c.NArg() == 0 {
|
if c.NArg() == 0 {
|
||||||
return fmt.Errorf("a list of pattern of files to extract is mandatory (e.g. '**' for all)")
|
return errors.New("a list of pattern of files to extract is mandatory (e.g. '**' for all)")
|
||||||
}
|
}
|
||||||
|
|
||||||
destdir := "."
|
destdir := "."
|
||||||
|
61
cmd/hook.go
61
cmd/hook.go
@ -220,10 +220,7 @@ Gitea or set your environment appropriately.`, "")
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
supportProcReceive := false
|
supportProcReceive := git.DefaultFeatures().SupportProcReceive
|
||||||
if git.CheckGitVersionAtLeast("2.29") == nil {
|
|
||||||
supportProcReceive = true
|
|
||||||
}
|
|
||||||
|
|
||||||
for scanner.Scan() {
|
for scanner.Scan() {
|
||||||
// TODO: support news feeds for wiki
|
// TODO: support news feeds for wiki
|
||||||
@ -293,8 +290,22 @@ Gitea or set your environment appropriately.`, "")
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// runHookUpdate avoid to do heavy operations on update hook because it will be
|
||||||
|
// invoked for every ref update which does not like pre-receive and post-receive
|
||||||
func runHookUpdate(c *cli.Context) error {
|
func runHookUpdate(c *cli.Context) error {
|
||||||
|
if isInternal, _ := strconv.ParseBool(os.Getenv(repo_module.EnvIsInternal)); isInternal {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// Update is empty and is kept only for backwards compatibility
|
// Update is empty and is kept only for backwards compatibility
|
||||||
|
if len(os.Args) < 3 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
refName := git.RefName(os.Args[len(os.Args)-3])
|
||||||
|
if refName.IsPull() {
|
||||||
|
// ignore update to refs/pull/xxx/head, so we don't need to output any information
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -341,6 +352,7 @@ Gitea or set your environment appropriately.`, "")
|
|||||||
isWiki, _ := strconv.ParseBool(os.Getenv(repo_module.EnvRepoIsWiki))
|
isWiki, _ := strconv.ParseBool(os.Getenv(repo_module.EnvRepoIsWiki))
|
||||||
repoName := os.Getenv(repo_module.EnvRepoName)
|
repoName := os.Getenv(repo_module.EnvRepoName)
|
||||||
pusherID, _ := strconv.ParseInt(os.Getenv(repo_module.EnvPusherID), 10, 64)
|
pusherID, _ := strconv.ParseInt(os.Getenv(repo_module.EnvPusherID), 10, 64)
|
||||||
|
prID, _ := strconv.ParseInt(os.Getenv(repo_module.EnvPRID), 10, 64)
|
||||||
pusherName := os.Getenv(repo_module.EnvPusherName)
|
pusherName := os.Getenv(repo_module.EnvPusherName)
|
||||||
|
|
||||||
hookOptions := private.HookOptions{
|
hookOptions := private.HookOptions{
|
||||||
@ -350,6 +362,8 @@ Gitea or set your environment appropriately.`, "")
|
|||||||
GitObjectDirectory: os.Getenv(private.GitObjectDirectory),
|
GitObjectDirectory: os.Getenv(private.GitObjectDirectory),
|
||||||
GitQuarantinePath: os.Getenv(private.GitQuarantinePath),
|
GitQuarantinePath: os.Getenv(private.GitQuarantinePath),
|
||||||
GitPushOptions: pushOptions(),
|
GitPushOptions: pushOptions(),
|
||||||
|
PullRequestID: prID,
|
||||||
|
PushTrigger: repo_module.PushTrigger(os.Getenv(repo_module.EnvPushTrigger)),
|
||||||
}
|
}
|
||||||
oldCommitIDs := make([]string, hookBatchSize)
|
oldCommitIDs := make([]string, hookBatchSize)
|
||||||
newCommitIDs := make([]string, hookBatchSize)
|
newCommitIDs := make([]string, hookBatchSize)
|
||||||
@ -448,23 +462,26 @@ Gitea or set your environment appropriately.`, "")
|
|||||||
|
|
||||||
func hookPrintResults(results []private.HookPostReceiveBranchResult) {
|
func hookPrintResults(results []private.HookPostReceiveBranchResult) {
|
||||||
for _, res := range results {
|
for _, res := range results {
|
||||||
if !res.Message {
|
hookPrintResult(res.Message, res.Create, res.Branch, res.URL)
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
fmt.Fprintln(os.Stderr, "")
|
|
||||||
if res.Create {
|
|
||||||
fmt.Fprintf(os.Stderr, "Create a new pull request for '%s':\n", res.Branch)
|
|
||||||
fmt.Fprintf(os.Stderr, " %s\n", res.URL)
|
|
||||||
} else {
|
|
||||||
fmt.Fprint(os.Stderr, "Visit the existing pull request:\n")
|
|
||||||
fmt.Fprintf(os.Stderr, " %s\n", res.URL)
|
|
||||||
}
|
|
||||||
fmt.Fprintln(os.Stderr, "")
|
|
||||||
os.Stderr.Sync()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func hookPrintResult(output, isCreate bool, branch, url string) {
|
||||||
|
if !output {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
fmt.Fprintln(os.Stderr, "")
|
||||||
|
if isCreate {
|
||||||
|
fmt.Fprintf(os.Stderr, "Create a new pull request for '%s':\n", branch)
|
||||||
|
fmt.Fprintf(os.Stderr, " %s\n", url)
|
||||||
|
} else {
|
||||||
|
fmt.Fprint(os.Stderr, "Visit the existing pull request:\n")
|
||||||
|
fmt.Fprintf(os.Stderr, " %s\n", url)
|
||||||
|
}
|
||||||
|
fmt.Fprintln(os.Stderr, "")
|
||||||
|
_ = os.Stderr.Sync()
|
||||||
|
}
|
||||||
|
|
||||||
func pushOptions() map[string]string {
|
func pushOptions() map[string]string {
|
||||||
opts := make(map[string]string)
|
opts := make(map[string]string)
|
||||||
if pushCount, err := strconv.Atoi(os.Getenv(private.GitPushOptionCount)); err == nil {
|
if pushCount, err := strconv.Atoi(os.Getenv(private.GitPushOptionCount)); err == nil {
|
||||||
@ -494,7 +511,7 @@ Gitea or set your environment appropriately.`, "")
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
if git.CheckGitVersionAtLeast("2.29") != nil {
|
if !git.DefaultFeatures().SupportProcReceive {
|
||||||
return fail(ctx, "No proc-receive support", "current git version doesn't support proc-receive.")
|
return fail(ctx, "No proc-receive support", "current git version doesn't support proc-receive.")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -691,6 +708,12 @@ Gitea or set your environment appropriately.`, "")
|
|||||||
}
|
}
|
||||||
err = writeFlushPktLine(ctx, os.Stdout)
|
err = writeFlushPktLine(ctx, os.Stdout)
|
||||||
|
|
||||||
|
if err == nil {
|
||||||
|
for _, res := range resp.Results {
|
||||||
|
hookPrintResult(res.ShouldShowMessage, res.IsCreatePR, res.HeadBranch, res.URL)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -112,13 +112,18 @@ func prepareWorkPathAndCustomConf(action cli.ActionFunc) func(ctx *cli.Context)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewMainApp(version, versionExtra string) *cli.App {
|
type AppVersion struct {
|
||||||
|
Version string
|
||||||
|
Extra string
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewMainApp(appVer AppVersion) *cli.App {
|
||||||
app := cli.NewApp()
|
app := cli.NewApp()
|
||||||
app.Name = "Gitea"
|
app.Name = "Gitea"
|
||||||
app.HelpName = "gitea"
|
app.HelpName = "gitea"
|
||||||
app.Usage = "A painless self-hosted Git service"
|
app.Usage = "A painless self-hosted Git service"
|
||||||
app.Description = `Gitea program contains "web" and other subcommands. If no subcommand is given, it starts the web server by default. Use "web" subcommand for more web server arguments, use other subcommands for other purposes.`
|
app.Description = `Gitea program contains "web" and other subcommands. If no subcommand is given, it starts the web server by default. Use "web" subcommand for more web server arguments, use other subcommands for other purposes.`
|
||||||
app.Version = version + versionExtra
|
app.Version = appVer.Version + appVer.Extra
|
||||||
app.EnableBashCompletion = true
|
app.EnableBashCompletion = true
|
||||||
|
|
||||||
// these sub-commands need to use config file
|
// these sub-commands need to use config file
|
||||||
|
@ -28,7 +28,7 @@ func makePathOutput(workPath, customPath, customConf string) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func newTestApp(testCmdAction func(ctx *cli.Context) error) *cli.App {
|
func newTestApp(testCmdAction func(ctx *cli.Context) error) *cli.App {
|
||||||
app := NewMainApp("version", "version-extra")
|
app := NewMainApp(AppVersion{})
|
||||||
testCmd := &cli.Command{Name: "test-cmd", Action: testCmdAction}
|
testCmd := &cli.Command{Name: "test-cmd", Action: testCmdAction}
|
||||||
prepareSubcommandWithConfig(testCmd, appGlobalFlags())
|
prepareSubcommandWithConfig(testCmd, appGlobalFlags())
|
||||||
app.Commands = append(app.Commands, testCmd)
|
app.Commands = append(app.Commands, testCmd)
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
package cmd
|
package cmd
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
@ -249,7 +250,7 @@ func runAddFileLogger(c *cli.Context) error {
|
|||||||
if c.IsSet("filename") {
|
if c.IsSet("filename") {
|
||||||
vals["filename"] = c.String("filename")
|
vals["filename"] = c.String("filename")
|
||||||
} else {
|
} else {
|
||||||
return fmt.Errorf("filename must be set when creating a file logger")
|
return errors.New("filename must be set when creating a file logger")
|
||||||
}
|
}
|
||||||
if c.IsSet("rotate") {
|
if c.IsSet("rotate") {
|
||||||
vals["rotate"] = c.Bool("rotate")
|
vals["rotate"] = c.Bool("rotate")
|
||||||
|
@ -5,7 +5,9 @@ package cmd
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io/fs"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
actions_model "code.gitea.io/gitea/models/actions"
|
actions_model "code.gitea.io/gitea/models/actions"
|
||||||
@ -34,13 +36,13 @@ var CmdMigrateStorage = &cli.Command{
|
|||||||
Name: "type",
|
Name: "type",
|
||||||
Aliases: []string{"t"},
|
Aliases: []string{"t"},
|
||||||
Value: "",
|
Value: "",
|
||||||
Usage: "Type of stored files to copy. Allowed types: 'attachments', 'lfs', 'avatars', 'repo-avatars', 'repo-archivers', 'packages', 'actions-log'",
|
Usage: "Type of stored files to copy. Allowed types: 'attachments', 'lfs', 'avatars', 'repo-avatars', 'repo-archivers', 'packages', 'actions-log', 'actions-artifacts",
|
||||||
},
|
},
|
||||||
&cli.StringFlag{
|
&cli.StringFlag{
|
||||||
Name: "storage",
|
Name: "storage",
|
||||||
Aliases: []string{"s"},
|
Aliases: []string{"s"},
|
||||||
Value: "",
|
Value: "",
|
||||||
Usage: "New storage type: local (default) or minio",
|
Usage: "New storage type: local (default), minio or azureblob",
|
||||||
},
|
},
|
||||||
&cli.StringFlag{
|
&cli.StringFlag{
|
||||||
Name: "path",
|
Name: "path",
|
||||||
@ -48,6 +50,7 @@ var CmdMigrateStorage = &cli.Command{
|
|||||||
Value: "",
|
Value: "",
|
||||||
Usage: "New storage placement if store is local (leave blank for default)",
|
Usage: "New storage placement if store is local (leave blank for default)",
|
||||||
},
|
},
|
||||||
|
// Minio Storage special configurations
|
||||||
&cli.StringFlag{
|
&cli.StringFlag{
|
||||||
Name: "minio-endpoint",
|
Name: "minio-endpoint",
|
||||||
Value: "",
|
Value: "",
|
||||||
@ -91,6 +94,37 @@ var CmdMigrateStorage = &cli.Command{
|
|||||||
Value: "",
|
Value: "",
|
||||||
Usage: "Minio checksum algorithm (default/md5)",
|
Usage: "Minio checksum algorithm (default/md5)",
|
||||||
},
|
},
|
||||||
|
&cli.StringFlag{
|
||||||
|
Name: "minio-bucket-lookup-type",
|
||||||
|
Value: "",
|
||||||
|
Usage: "Minio bucket lookup type",
|
||||||
|
},
|
||||||
|
// Azure Blob Storage special configurations
|
||||||
|
&cli.StringFlag{
|
||||||
|
Name: "azureblob-endpoint",
|
||||||
|
Value: "",
|
||||||
|
Usage: "Azure Blob storage endpoint",
|
||||||
|
},
|
||||||
|
&cli.StringFlag{
|
||||||
|
Name: "azureblob-account-name",
|
||||||
|
Value: "",
|
||||||
|
Usage: "Azure Blob storage account name",
|
||||||
|
},
|
||||||
|
&cli.StringFlag{
|
||||||
|
Name: "azureblob-account-key",
|
||||||
|
Value: "",
|
||||||
|
Usage: "Azure Blob storage account key",
|
||||||
|
},
|
||||||
|
&cli.StringFlag{
|
||||||
|
Name: "azureblob-container",
|
||||||
|
Value: "",
|
||||||
|
Usage: "Azure Blob storage container",
|
||||||
|
},
|
||||||
|
&cli.StringFlag{
|
||||||
|
Name: "azureblob-base-path",
|
||||||
|
Value: "",
|
||||||
|
Usage: "Azure Blob storage base path",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -160,6 +194,25 @@ func migrateActionsLog(ctx context.Context, dstStorage storage.ObjectStorage) er
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func migrateActionsArtifacts(ctx context.Context, dstStorage storage.ObjectStorage) error {
|
||||||
|
return db.Iterate(ctx, nil, func(ctx context.Context, artifact *actions_model.ActionArtifact) error {
|
||||||
|
if artifact.Status == int64(actions_model.ArtifactStatusExpired) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := storage.Copy(dstStorage, artifact.StoragePath, storage.ActionsArtifacts, artifact.StoragePath)
|
||||||
|
if err != nil {
|
||||||
|
// ignore files that do not exist
|
||||||
|
if errors.Is(err, fs.ErrNotExist) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func runMigrateStorage(ctx *cli.Context) error {
|
func runMigrateStorage(ctx *cli.Context) error {
|
||||||
stdCtx, cancel := installSignals()
|
stdCtx, cancel := installSignals()
|
||||||
defer cancel()
|
defer cancel()
|
||||||
@ -213,6 +266,19 @@ func runMigrateStorage(ctx *cli.Context) error {
|
|||||||
UseSSL: ctx.Bool("minio-use-ssl"),
|
UseSSL: ctx.Bool("minio-use-ssl"),
|
||||||
InsecureSkipVerify: ctx.Bool("minio-insecure-skip-verify"),
|
InsecureSkipVerify: ctx.Bool("minio-insecure-skip-verify"),
|
||||||
ChecksumAlgorithm: ctx.String("minio-checksum-algorithm"),
|
ChecksumAlgorithm: ctx.String("minio-checksum-algorithm"),
|
||||||
|
BucketLookUpType: ctx.String("minio-bucket-lookup-type"),
|
||||||
|
},
|
||||||
|
})
|
||||||
|
case string(setting.AzureBlobStorageType):
|
||||||
|
dstStorage, err = storage.NewAzureBlobStorage(
|
||||||
|
stdCtx,
|
||||||
|
&setting.Storage{
|
||||||
|
AzureBlobConfig: setting.AzureBlobStorageConfig{
|
||||||
|
Endpoint: ctx.String("azureblob-endpoint"),
|
||||||
|
AccountName: ctx.String("azureblob-account-name"),
|
||||||
|
AccountKey: ctx.String("azureblob-account-key"),
|
||||||
|
Container: ctx.String("azureblob-container"),
|
||||||
|
BasePath: ctx.String("azureblob-base-path"),
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
default:
|
default:
|
||||||
@ -223,13 +289,14 @@ func runMigrateStorage(ctx *cli.Context) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
migratedMethods := map[string]func(context.Context, storage.ObjectStorage) error{
|
migratedMethods := map[string]func(context.Context, storage.ObjectStorage) error{
|
||||||
"attachments": migrateAttachments,
|
"attachments": migrateAttachments,
|
||||||
"lfs": migrateLFS,
|
"lfs": migrateLFS,
|
||||||
"avatars": migrateAvatars,
|
"avatars": migrateAvatars,
|
||||||
"repo-avatars": migrateRepoAvatars,
|
"repo-avatars": migrateRepoAvatars,
|
||||||
"repo-archivers": migrateRepoArchivers,
|
"repo-archivers": migrateRepoArchivers,
|
||||||
"packages": migratePackages,
|
"packages": migratePackages,
|
||||||
"actions-log": migrateActionsLog,
|
"actions-log": migrateActionsLog,
|
||||||
|
"actions-artifacts": migrateActionsArtifacts,
|
||||||
}
|
}
|
||||||
|
|
||||||
tp := strings.ToLower(ctx.String("type"))
|
tp := strings.ToLower(ctx.String("type"))
|
||||||
|
@ -178,7 +178,7 @@ func runServ(c *cli.Context) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if len(words) < 2 {
|
if len(words) < 2 {
|
||||||
if git.CheckGitVersionAtLeast("2.29") == nil {
|
if git.DefaultFeatures().SupportProcReceive {
|
||||||
// for AGit Flow
|
// for AGit Flow
|
||||||
if cmd == "ssh_info" {
|
if cmd == "ssh_info" {
|
||||||
fmt.Print(`{"type":"gitea","version":1}`)
|
fmt.Print(`{"type":"gitea","version":1}`)
|
||||||
|
@ -114,7 +114,7 @@ func showWebStartupMessage(msg string) {
|
|||||||
log.Info("* WorkPath: %s", setting.AppWorkPath)
|
log.Info("* WorkPath: %s", setting.AppWorkPath)
|
||||||
log.Info("* CustomPath: %s", setting.CustomPath)
|
log.Info("* CustomPath: %s", setting.CustomPath)
|
||||||
log.Info("* ConfigFile: %s", setting.CustomConf)
|
log.Info("* ConfigFile: %s", setting.CustomConf)
|
||||||
log.Info("%s", msg)
|
log.Info("%s", msg) // show startup message
|
||||||
}
|
}
|
||||||
|
|
||||||
func serveInstall(ctx *cli.Context) error {
|
func serveInstall(ctx *cli.Context) error {
|
||||||
|
@ -17,7 +17,7 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
|
||||||
"github.com/google/go-github/v57/github"
|
"github.com/google/go-github/v61/github"
|
||||||
"github.com/urfave/cli/v2"
|
"github.com/urfave/cli/v2"
|
||||||
"gopkg.in/yaml.v3"
|
"gopkg.in/yaml.v3"
|
||||||
)
|
)
|
||||||
|
12
crowdin.yml
Normal file
12
crowdin.yml
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
project_id_env: CROWDIN_PROJECT_ID
|
||||||
|
api_token_env: CROWDIN_KEY
|
||||||
|
base_path: "."
|
||||||
|
base_url: "https://api.crowdin.com"
|
||||||
|
preserve_hierarchy: true
|
||||||
|
files:
|
||||||
|
- source: "/options/locale/locale_en-US.ini"
|
||||||
|
translation: "/options/locale/locale_%locale%.ini"
|
||||||
|
type: "ini"
|
||||||
|
skip_untranslated_strings: true
|
||||||
|
export_only_approved: true
|
||||||
|
update_option: "update_as_unapproved"
|
@ -81,6 +81,10 @@ RUN_USER = ; git
|
|||||||
;; Overwrite the automatically generated public URL. Necessary for proxies and docker.
|
;; Overwrite the automatically generated public URL. Necessary for proxies and docker.
|
||||||
;ROOT_URL = %(PROTOCOL)s://%(DOMAIN)s:%(HTTP_PORT)s/
|
;ROOT_URL = %(PROTOCOL)s://%(DOMAIN)s:%(HTTP_PORT)s/
|
||||||
;;
|
;;
|
||||||
|
;; For development purpose only. It makes Gitea handle sub-path ("/sub-path/owner/repo/...") directly when debugging without a reverse proxy.
|
||||||
|
;; DO NOT USE IT IN PRODUCTION!!!
|
||||||
|
;USE_SUB_URL_PATH = false
|
||||||
|
;;
|
||||||
;; when STATIC_URL_PREFIX is empty it will follow ROOT_URL
|
;; when STATIC_URL_PREFIX is empty it will follow ROOT_URL
|
||||||
;STATIC_URL_PREFIX =
|
;STATIC_URL_PREFIX =
|
||||||
;;
|
;;
|
||||||
@ -441,7 +445,7 @@ INTERNAL_TOKEN =
|
|||||||
;INTERNAL_TOKEN_URI = file:/etc/gitea/internal_token
|
;INTERNAL_TOKEN_URI = file:/etc/gitea/internal_token
|
||||||
;;
|
;;
|
||||||
;; How long to remember that a user is logged in before requiring relogin (in days)
|
;; How long to remember that a user is logged in before requiring relogin (in days)
|
||||||
;LOGIN_REMEMBER_DAYS = 7
|
;LOGIN_REMEMBER_DAYS = 31
|
||||||
;;
|
;;
|
||||||
;; Name of the cookie used to store the current username.
|
;; Name of the cookie used to store the current username.
|
||||||
;COOKIE_USERNAME = gitea_awesome
|
;COOKIE_USERNAME = gitea_awesome
|
||||||
@ -1231,7 +1235,8 @@ LEVEL = Info
|
|||||||
;DEFAULT_THEME = gitea-auto
|
;DEFAULT_THEME = gitea-auto
|
||||||
;;
|
;;
|
||||||
;; All available themes. Allow users select personalized themes regardless of the value of `DEFAULT_THEME`.
|
;; All available themes. Allow users select personalized themes regardless of the value of `DEFAULT_THEME`.
|
||||||
;THEMES = gitea-auto,gitea-light,gitea-dark
|
;; Leave it empty to allow users to select any theme from "{CustomPath}/public/assets/css/theme-*.css"
|
||||||
|
;THEMES =
|
||||||
;;
|
;;
|
||||||
;; All available reactions users can choose on issues/prs and comments.
|
;; All available reactions users can choose on issues/prs and comments.
|
||||||
;; Values can be emoji alias (:smile:) or a unicode emoji.
|
;; Values can be emoji alias (:smile:) or a unicode emoji.
|
||||||
@ -1333,6 +1338,9 @@ LEVEL = Info
|
|||||||
;;
|
;;
|
||||||
;; Maximum allowed file size in bytes to render CSV files as table. (Set to 0 for no limit).
|
;; Maximum allowed file size in bytes to render CSV files as table. (Set to 0 for no limit).
|
||||||
;MAX_FILE_SIZE = 524288
|
;MAX_FILE_SIZE = 524288
|
||||||
|
;;
|
||||||
|
;; Maximum allowed rows to render CSV files. (Set to 0 for no limit)
|
||||||
|
;MAX_ROWS = 2500
|
||||||
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
@ -1455,7 +1463,7 @@ LEVEL = Info
|
|||||||
;; Batch size to send for batched queues
|
;; Batch size to send for batched queues
|
||||||
;BATCH_LENGTH = 20
|
;BATCH_LENGTH = 20
|
||||||
;;
|
;;
|
||||||
;; Connection string for redis queues this will store the redis or redis-cluster connection string.
|
;; Connection string for redis queues this will store the redis (or Redis cluster) connection string.
|
||||||
;; When `TYPE` is `persistable-channel`, this provides a directory for the underlying leveldb
|
;; When `TYPE` is `persistable-channel`, this provides a directory for the underlying leveldb
|
||||||
;; or additional options of the form `leveldb://path/to/db?option=value&....`, and will override `DATADIR`.
|
;; or additional options of the form `leveldb://path/to/db?option=value&....`, and will override `DATADIR`.
|
||||||
;CONN_STR = "redis://127.0.0.1:6379/0"
|
;CONN_STR = "redis://127.0.0.1:6379/0"
|
||||||
@ -1480,11 +1488,20 @@ LEVEL = Info
|
|||||||
;;
|
;;
|
||||||
;; Default configuration for email notifications for users (user configurable). Options: enabled, onmention, disabled
|
;; Default configuration for email notifications for users (user configurable). Options: enabled, onmention, disabled
|
||||||
;DEFAULT_EMAIL_NOTIFICATIONS = enabled
|
;DEFAULT_EMAIL_NOTIFICATIONS = enabled
|
||||||
;; Disabled features for users, could be "deletion", "manage_ssh_keys","manage_gpg_keys" more features can be disabled in future
|
;; Disabled features for users could be "deletion", "manage_ssh_keys", "manage_gpg_keys", "manage_mfa", "manage_credentials" more features can be disabled in future
|
||||||
;; - deletion: a user cannot delete their own account
|
;; - deletion: a user cannot delete their own account
|
||||||
;; - manage_ssh_keys: a user cannot configure ssh keys
|
;; - manage_ssh_keys: a user cannot configure ssh keys
|
||||||
;; - manage_gpg_keys: a user cannot configure gpg keys
|
;; - manage_gpg_keys: a user cannot configure gpg keys
|
||||||
|
;; - manage_mfa: a user cannot configure mfa devices
|
||||||
|
;; - manage_credentials: a user cannot configure emails, passwords, or openid
|
||||||
;USER_DISABLED_FEATURES =
|
;USER_DISABLED_FEATURES =
|
||||||
|
;; Comma separated list of disabled features ONLY if the user has an external login type (eg. LDAP, Oauth, etc.), could be "deletion", "manage_ssh_keys", "manage_gpg_keys", "manage_mfa", "manage_credentials". This setting is independent from `USER_DISABLED_FEATURES` and supplements its behavior.
|
||||||
|
;; - deletion: a user cannot delete their own account
|
||||||
|
;; - manage_ssh_keys: a user cannot configure ssh keys
|
||||||
|
;; - manage_gpg_keys: a user cannot configure gpg keys
|
||||||
|
;; - manage_mfa: a user cannot configure mfa devices
|
||||||
|
;; - manage_credentials: a user cannot configure emails, passwords, or openid
|
||||||
|
;;EXTERNAL_USER_DISABLE_FEATURES =
|
||||||
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
@ -1548,11 +1565,12 @@ LEVEL = Info
|
|||||||
;; The source of the username for new oauth2 accounts:
|
;; The source of the username for new oauth2 accounts:
|
||||||
;; userid = use the userid / sub attribute
|
;; userid = use the userid / sub attribute
|
||||||
;; nickname = use the nickname attribute
|
;; nickname = use the nickname attribute
|
||||||
|
;; preferred_username = use the preferred_username attribute
|
||||||
;; email = use the username part of the email attribute
|
;; email = use the username part of the email attribute
|
||||||
;; Note: `nickname` and `email` options will normalize input strings using the following criteria:
|
;; Note: `nickname`, `preferred_username` and `email` options will normalize input strings using the following criteria:
|
||||||
;; - diacritics are removed
|
;; - diacritics are removed
|
||||||
;; - the characters in the set `['´\x60]` are removed
|
;; - the characters in the set ['´`] are removed
|
||||||
;; - the characters in the set `[\s~+]` are replaced with `-`
|
;; - the characters in the set [\s~+] are replaced with "-"
|
||||||
;USERNAME = nickname
|
;USERNAME = nickname
|
||||||
;;
|
;;
|
||||||
;; Update avatar if available from oauth2 provider.
|
;; Update avatar if available from oauth2 provider.
|
||||||
@ -1658,6 +1676,10 @@ LEVEL = Info
|
|||||||
;; Sometimes it is helpful to use a different address on the envelope. Set this to use ENVELOPE_FROM as the from on the envelope. Set to `<>` to send an empty address.
|
;; Sometimes it is helpful to use a different address on the envelope. Set this to use ENVELOPE_FROM as the from on the envelope. Set to `<>` to send an empty address.
|
||||||
;ENVELOPE_FROM =
|
;ENVELOPE_FROM =
|
||||||
;;
|
;;
|
||||||
|
;; If gitea sends mails on behave of users, it will just use the name also displayed in the WebUI. If you want e.g. `Mister X (by CodeIt) <gitea@codeit.net>`,
|
||||||
|
;; set it to `{{ .DisplayName }} (by {{ .AppName }})`. Available Variables: `.DisplayName`, `.AppName` and `.Domain`.
|
||||||
|
;FROM_DISPLAY_NAME_FORMAT = {{ .DisplayName }}
|
||||||
|
;;
|
||||||
;; Mailer user name and password, if required by provider.
|
;; Mailer user name and password, if required by provider.
|
||||||
;USER =
|
;USER =
|
||||||
;;
|
;;
|
||||||
@ -1680,6 +1702,16 @@ LEVEL = Info
|
|||||||
;; convert \r\n to \n for Sendmail
|
;; convert \r\n to \n for Sendmail
|
||||||
;SENDMAIL_CONVERT_CRLF = true
|
;SENDMAIL_CONVERT_CRLF = true
|
||||||
|
|
||||||
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
|
;[mailer.override_header]
|
||||||
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
|
;; This is empty by default, use it only if you know what you need it for.
|
||||||
|
;Reply-To = test@example.com, test2@example.com
|
||||||
|
;Content-Type = text/html; charset=utf-8
|
||||||
|
;In-Reply-To =
|
||||||
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;[email.incoming]
|
;[email.incoming]
|
||||||
@ -1733,9 +1765,8 @@ LEVEL = Info
|
|||||||
;; For "memory" only, GC interval in seconds, default is 60
|
;; For "memory" only, GC interval in seconds, default is 60
|
||||||
;INTERVAL = 60
|
;INTERVAL = 60
|
||||||
;;
|
;;
|
||||||
;; For "redis", "redis-cluster" and "memcache", connection host address
|
;; For "redis" and "memcache", connection host address
|
||||||
;; redis: `redis://127.0.0.1:6379/0?pool_size=100&idle_timeout=180s`
|
;; redis: `redis://127.0.0.1:6379/0?pool_size=100&idle_timeout=180s` (or `redis+cluster://127.0.0.1:6379/0?pool_size=100&idle_timeout=180s` for a Redis cluster)
|
||||||
;; redis-cluster: `redis+cluster://127.0.0.1:6379/0?pool_size=100&idle_timeout=180s`
|
|
||||||
;; memcache: `127.0.0.1:11211`
|
;; memcache: `127.0.0.1:11211`
|
||||||
;; twoqueue: `{"size":50000,"recent_ratio":0.25,"ghost_ratio":0.5}` or `50000`
|
;; twoqueue: `{"size":50000,"recent_ratio":0.25,"ghost_ratio":0.5}` or `50000`
|
||||||
;HOST =
|
;HOST =
|
||||||
@ -1765,15 +1796,14 @@ LEVEL = Info
|
|||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;
|
;;
|
||||||
;; Either "memory", "file", "redis", "redis-cluster", "db", "mysql", "couchbase", "memcache" or "postgres"
|
;; Either "memory", "file", "redis", "db", "mysql", "couchbase", "memcache" or "postgres"
|
||||||
;; Default is "memory". "db" will reuse the configuration in [database]
|
;; Default is "memory". "db" will reuse the configuration in [database]
|
||||||
;PROVIDER = memory
|
;PROVIDER = memory
|
||||||
;;
|
;;
|
||||||
;; Provider config options
|
;; Provider config options
|
||||||
;; memory: doesn't have any config yet
|
;; memory: doesn't have any config yet
|
||||||
;; file: session file path, e.g. `data/sessions`
|
;; file: session file path, e.g. `data/sessions`
|
||||||
;; redis: `redis://127.0.0.1:6379/0?pool_size=100&idle_timeout=180s`
|
;; redis: `redis://127.0.0.1:6379/0?pool_size=100&idle_timeout=180s` (or `redis+cluster://127.0.0.1:6379/0?pool_size=100&idle_timeout=180s` for a Redis cluster)
|
||||||
;; redis-cluster: `redis+cluster://127.0.0.1:6379/0?pool_size=100&idle_timeout=180s`
|
|
||||||
;; mysql: go-sql-driver/mysql dsn config string, e.g. `root:password@/session_table`
|
;; mysql: go-sql-driver/mysql dsn config string, e.g. `root:password@/session_table`
|
||||||
;PROVIDER_CONFIG = data/sessions ; Relative paths will be made absolute against _`AppWorkPath`_.
|
;PROVIDER_CONFIG = data/sessions ; Relative paths will be made absolute against _`AppWorkPath`_.
|
||||||
;;
|
;;
|
||||||
@ -1857,7 +1887,7 @@ LEVEL = Info
|
|||||||
;STORAGE_TYPE = local
|
;STORAGE_TYPE = local
|
||||||
;;
|
;;
|
||||||
;; Allows the storage driver to redirect to authenticated URLs to serve files directly
|
;; Allows the storage driver to redirect to authenticated URLs to serve files directly
|
||||||
;; Currently, only `minio` is supported.
|
;; Currently, only `minio` and `azureblob` is supported.
|
||||||
;SERVE_DIRECT = false
|
;SERVE_DIRECT = false
|
||||||
;;
|
;;
|
||||||
;; Path for attachments. Defaults to `attachments`. Only available when STORAGE_TYPE is `local`
|
;; Path for attachments. Defaults to `attachments`. Only available when STORAGE_TYPE is `local`
|
||||||
@ -1867,7 +1897,10 @@ LEVEL = Info
|
|||||||
;; Minio endpoint to connect only available when STORAGE_TYPE is `minio`
|
;; Minio endpoint to connect only available when STORAGE_TYPE is `minio`
|
||||||
;MINIO_ENDPOINT = localhost:9000
|
;MINIO_ENDPOINT = localhost:9000
|
||||||
;;
|
;;
|
||||||
;; Minio accessKeyID to connect only available when STORAGE_TYPE is `minio`
|
;; Minio accessKeyID to connect only available when STORAGE_TYPE is `minio`.
|
||||||
|
;; If not provided and STORAGE_TYPE is `minio`, will search for credentials in known
|
||||||
|
;; environment variables (MINIO_ACCESS_KEY_ID, AWS_ACCESS_KEY_ID), credentials files
|
||||||
|
;; (~/.mc/config.json, ~/.aws/credentials), and EC2 instance metadata.
|
||||||
;MINIO_ACCESS_KEY_ID =
|
;MINIO_ACCESS_KEY_ID =
|
||||||
;;
|
;;
|
||||||
;; Minio secretAccessKey to connect only available when STORAGE_TYPE is `minio`
|
;; Minio secretAccessKey to connect only available when STORAGE_TYPE is `minio`
|
||||||
@ -1890,6 +1923,24 @@ LEVEL = Info
|
|||||||
;;
|
;;
|
||||||
;; Minio checksum algorithm: default (for MinIO or AWS S3) or md5 (for Cloudflare or Backblaze)
|
;; Minio checksum algorithm: default (for MinIO or AWS S3) or md5 (for Cloudflare or Backblaze)
|
||||||
;MINIO_CHECKSUM_ALGORITHM = default
|
;MINIO_CHECKSUM_ALGORITHM = default
|
||||||
|
;;
|
||||||
|
;; Minio bucket lookup method defaults to auto mode; set it to `dns` for virtual host style or `path` for path style, only available when STORAGE_TYPE is `minio`
|
||||||
|
;MINIO_BUCKET_LOOKUP_TYPE = auto
|
||||||
|
;; Azure Blob endpoint to connect only available when STORAGE_TYPE is `azureblob`,
|
||||||
|
;; e.g. https://accountname.blob.core.windows.net or http://127.0.0.1:10000/devstoreaccount1
|
||||||
|
;AZURE_BLOB_ENDPOINT =
|
||||||
|
;;
|
||||||
|
;; Azure Blob account name to connect only available when STORAGE_TYPE is `azureblob`
|
||||||
|
;AZURE_BLOB_ACCOUNT_NAME =
|
||||||
|
;;
|
||||||
|
;; Azure Blob account key to connect only available when STORAGE_TYPE is `azureblob`
|
||||||
|
;AZURE_BLOB_ACCOUNT_KEY =
|
||||||
|
;;
|
||||||
|
;; Azure Blob container to store the attachments only available when STORAGE_TYPE is `azureblob`
|
||||||
|
;AZURE_BLOB_CONTAINER = gitea
|
||||||
|
;;
|
||||||
|
;; override the azure blob base path if storage type is azureblob
|
||||||
|
;AZURE_BLOB_BASE_PATH = attachments/
|
||||||
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
@ -2028,6 +2079,17 @@ LEVEL = Info
|
|||||||
;; or only create new users if UPDATE_EXISTING is set to false
|
;; or only create new users if UPDATE_EXISTING is set to false
|
||||||
;UPDATE_EXISTING = true
|
;UPDATE_EXISTING = true
|
||||||
|
|
||||||
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
|
;; Cleanup expired actions assets
|
||||||
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
|
;[cron.cleanup_actions]
|
||||||
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
|
;ENABLED = true
|
||||||
|
;RUN_AT_START = true
|
||||||
|
;SCHEDULE = @midnight
|
||||||
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;; Clean-up deleted branches
|
;; Clean-up deleted branches
|
||||||
@ -2310,6 +2372,8 @@ LEVEL = Info
|
|||||||
;SHOW_FOOTER_VERSION = true
|
;SHOW_FOOTER_VERSION = true
|
||||||
;; Show template execution time in the footer
|
;; Show template execution time in the footer
|
||||||
;SHOW_FOOTER_TEMPLATE_LOAD_TIME = true
|
;SHOW_FOOTER_TEMPLATE_LOAD_TIME = true
|
||||||
|
;; Show the "powered by" text in the footer
|
||||||
|
;SHOW_FOOTER_POWERED_BY = true
|
||||||
;; Generate sitemap. Defaults to `true`.
|
;; Generate sitemap. Defaults to `true`.
|
||||||
;ENABLE_SITEMAP = true
|
;ENABLE_SITEMAP = true
|
||||||
;; Enable/Disable RSS/Atom feed
|
;; Enable/Disable RSS/Atom feed
|
||||||
@ -2369,22 +2433,6 @@ LEVEL = Info
|
|||||||
;; Enable issue by repository metrics; default is false
|
;; Enable issue by repository metrics; default is false
|
||||||
;ENABLED_ISSUE_BY_REPOSITORY = false
|
;ENABLED_ISSUE_BY_REPOSITORY = false
|
||||||
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
||||||
;[task]
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
||||||
;;
|
|
||||||
;; Task queue type, could be `channel` or `redis`.
|
|
||||||
;QUEUE_TYPE = channel
|
|
||||||
;;
|
|
||||||
;; Task queue length, available only when `QUEUE_TYPE` is `channel`.
|
|
||||||
;QUEUE_LENGTH = 1000
|
|
||||||
;;
|
|
||||||
;; Task queue connection string, available only when `QUEUE_TYPE` is `redis`.
|
|
||||||
;; If there is a password of redis, use `redis://127.0.0.1:6379/0?pool_size=100&idle_timeout=180s` or `redis+cluster://127.0.0.1:6379/0?pool_size=100&idle_timeout=180s` for `redis-clsuter`.
|
|
||||||
;QUEUE_CONN_STR = "redis://127.0.0.1:6379/0?pool_size=100&idle_timeout=180s"
|
|
||||||
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;[migrations]
|
;[migrations]
|
||||||
@ -2452,6 +2500,11 @@ LEVEL = Info
|
|||||||
;STORAGE_TYPE = local
|
;STORAGE_TYPE = local
|
||||||
;; override the minio base path if storage type is minio
|
;; override the minio base path if storage type is minio
|
||||||
;MINIO_BASE_PATH = packages/
|
;MINIO_BASE_PATH = packages/
|
||||||
|
;; override the azure blob base path if storage type is azureblob
|
||||||
|
;AZURE_BLOB_BASE_PATH = packages/
|
||||||
|
;; Allows the storage driver to redirect to authenticated URLs to serve files directly
|
||||||
|
;; Currently, only `minio` and `azureblob` is supported.
|
||||||
|
;SERVE_DIRECT = false
|
||||||
;;
|
;;
|
||||||
;; Path for chunked uploads. Defaults to APP_DATA_PATH + `tmp/package-upload`
|
;; Path for chunked uploads. Defaults to APP_DATA_PATH + `tmp/package-upload`
|
||||||
;CHUNKED_UPLOAD_PATH = tmp/package-upload
|
;CHUNKED_UPLOAD_PATH = tmp/package-upload
|
||||||
@ -2502,7 +2555,8 @@ LEVEL = Info
|
|||||||
;LIMIT_SIZE_SWIFT = -1
|
;LIMIT_SIZE_SWIFT = -1
|
||||||
;; Maximum size of a Vagrant upload (`-1` means no limits, format `1000`, `1 MB`, `1 GiB`)
|
;; Maximum size of a Vagrant upload (`-1` means no limits, format `1000`, `1 MB`, `1 GiB`)
|
||||||
;LIMIT_SIZE_VAGRANT = -1
|
;LIMIT_SIZE_VAGRANT = -1
|
||||||
|
;; Enable RPM re-signing by default. (It will overwrite the old signature ,using v4 format, not compatible with CentOS 6 or older)
|
||||||
|
;DEFAULT_RPM_SIGN_ENABLED = false
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;; default storage for attachments, lfs and avatars
|
;; default storage for attachments, lfs and avatars
|
||||||
@ -2525,6 +2579,8 @@ LEVEL = Info
|
|||||||
;;
|
;;
|
||||||
;; override the minio base path if storage type is minio
|
;; override the minio base path if storage type is minio
|
||||||
;MINIO_BASE_PATH = repo-archive/
|
;MINIO_BASE_PATH = repo-archive/
|
||||||
|
;; override the azure blob base path if storage type is azureblob
|
||||||
|
;AZURE_BLOB_BASE_PATH = repo-archive/
|
||||||
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
@ -2546,8 +2602,15 @@ LEVEL = Info
|
|||||||
;; Where your lfs files reside, default is data/lfs.
|
;; Where your lfs files reside, default is data/lfs.
|
||||||
;PATH = data/lfs
|
;PATH = data/lfs
|
||||||
;;
|
;;
|
||||||
|
;; Allows the storage driver to redirect to authenticated URLs to serve files directly
|
||||||
|
;; Currently, only `minio` and `azureblob` is supported.
|
||||||
|
;SERVE_DIRECT = false
|
||||||
|
;;
|
||||||
;; override the minio base path if storage type is minio
|
;; override the minio base path if storage type is minio
|
||||||
;MINIO_BASE_PATH = lfs/
|
;MINIO_BASE_PATH = lfs/
|
||||||
|
;;
|
||||||
|
;; override the azure blob base path if storage type is azureblob
|
||||||
|
;AZURE_BLOB_BASE_PATH = lfs/
|
||||||
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
@ -2562,13 +2625,16 @@ LEVEL = Info
|
|||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;; customize storage
|
;; customize storage
|
||||||
;[storage.my_minio]
|
;[storage.minio]
|
||||||
;STORAGE_TYPE = minio
|
;STORAGE_TYPE = minio
|
||||||
;;
|
;;
|
||||||
;; Minio endpoint to connect only available when STORAGE_TYPE is `minio`
|
;; Minio endpoint to connect only available when STORAGE_TYPE is `minio`
|
||||||
;MINIO_ENDPOINT = localhost:9000
|
;MINIO_ENDPOINT = localhost:9000
|
||||||
;;
|
;;
|
||||||
;; Minio accessKeyID to connect only available when STORAGE_TYPE is `minio`
|
;; Minio accessKeyID to connect only available when STORAGE_TYPE is `minio`.
|
||||||
|
;; If not provided and STORAGE_TYPE is `minio`, will search for credentials in known
|
||||||
|
;; environment variables (MINIO_ACCESS_KEY_ID, AWS_ACCESS_KEY_ID), credentials files
|
||||||
|
;; (~/.mc/config.json, ~/.aws/credentials), and EC2 instance metadata.
|
||||||
;MINIO_ACCESS_KEY_ID =
|
;MINIO_ACCESS_KEY_ID =
|
||||||
;;
|
;;
|
||||||
;; Minio secretAccessKey to connect only available when STORAGE_TYPE is `minio`
|
;; Minio secretAccessKey to connect only available when STORAGE_TYPE is `minio`
|
||||||
@ -2585,6 +2651,25 @@ LEVEL = Info
|
|||||||
;;
|
;;
|
||||||
;; Minio skip SSL verification available when STORAGE_TYPE is `minio`
|
;; Minio skip SSL verification available when STORAGE_TYPE is `minio`
|
||||||
;MINIO_INSECURE_SKIP_VERIFY = false
|
;MINIO_INSECURE_SKIP_VERIFY = false
|
||||||
|
;;
|
||||||
|
;; Minio bucket lookup method defaults to auto mode; set it to `dns` for virtual host style or `path` for path style, only available when STORAGE_TYPE is `minio`
|
||||||
|
;MINIO_BUCKET_LOOKUP_TYPE = auto
|
||||||
|
|
||||||
|
;[storage.azureblob]
|
||||||
|
;STORAGE_TYPE = azureblob
|
||||||
|
;;
|
||||||
|
;; Azure Blob endpoint to connect only available when STORAGE_TYPE is `azureblob`,
|
||||||
|
;; e.g. https://accountname.blob.core.windows.net or http://127.0.0.1:10000/devstoreaccount1
|
||||||
|
;AZURE_BLOB_ENDPOINT =
|
||||||
|
;;
|
||||||
|
;; Azure Blob account name to connect only available when STORAGE_TYPE is `azureblob`
|
||||||
|
;AZURE_BLOB_ACCOUNT_NAME =
|
||||||
|
;;
|
||||||
|
;; Azure Blob account key to connect only available when STORAGE_TYPE is `azureblob`
|
||||||
|
;AZURE_BLOB_ACCOUNT_KEY =
|
||||||
|
;;
|
||||||
|
;; Azure Blob container to store the attachments only available when STORAGE_TYPE is `azureblob`
|
||||||
|
;AZURE_BLOB_CONTAINER = gitea
|
||||||
|
|
||||||
;[proxy]
|
;[proxy]
|
||||||
;; Enable the proxy, all requests to external via HTTP will be affected
|
;; Enable the proxy, all requests to external via HTTP will be affected
|
||||||
@ -2600,6 +2685,14 @@ LEVEL = Info
|
|||||||
;;
|
;;
|
||||||
;; Default platform to get action plugins, `github` for `https://github.com`, `self` for the current Gitea instance.
|
;; Default platform to get action plugins, `github` for `https://github.com`, `self` for the current Gitea instance.
|
||||||
;DEFAULT_ACTIONS_URL = github
|
;DEFAULT_ACTIONS_URL = github
|
||||||
|
;; Logs retention time in days. Old logs will be deleted after this period.
|
||||||
|
;LOG_RETENTION_DAYS = 365
|
||||||
|
;; Log compression type, `none` for no compression, `zstd` for zstd compression.
|
||||||
|
;; Other compression types like `gzip` are NOT supported, since seekable stream is required for log view.
|
||||||
|
;; It's always recommended to use compression when using local disk as log storage if CPU or memory is not a bottleneck.
|
||||||
|
;; And for object storage services like S3, which is billed for requests, it would cause extra 2 times of get requests for each log view.
|
||||||
|
;; But it will save storage space and network bandwidth, so it's still recommended to use compression.
|
||||||
|
;LOG_COMPRESSION = none
|
||||||
;; Default artifact retention time in days. Artifacts could have their own retention periods by setting the `retention-days` option in `actions/upload-artifact` step.
|
;; Default artifact retention time in days. Artifacts could have their own retention periods by setting the `retention-days` option in `actions/upload-artifact` step.
|
||||||
;ARTIFACT_RETENTION_DAYS = 90
|
;ARTIFACT_RETENTION_DAYS = 90
|
||||||
;; Timeout to stop the task which have running status, but haven't been updated for a long time
|
;; Timeout to stop the task which have running status, but haven't been updated for a long time
|
||||||
@ -2620,3 +2713,9 @@ LEVEL = Info
|
|||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;; storage type
|
;; storage type
|
||||||
;STORAGE_TYPE = local
|
;STORAGE_TYPE = local
|
||||||
|
|
||||||
|
;[global_lock]
|
||||||
|
;; Lock service type, could be memory or redis
|
||||||
|
;SERVICE_TYPE = memory
|
||||||
|
;; Ignored for the "memory" type. For "redis" use something like `redis://127.0.0.1:6379/0`
|
||||||
|
;SERVICE_CONN_STR =
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
# Gitea - Docker
|
# Gitea - Docker
|
||||||
|
|
||||||
Dockerfile is found in root of repository.
|
Dockerfile is found in the root of the repository.
|
||||||
|
|
||||||
Docker image can be found on [docker hub](https://hub.docker.com/r/gitea/gitea)
|
Docker image can be found on [docker hub](https://hub.docker.com/r/gitea/gitea).
|
||||||
|
|
||||||
Documentation on using docker image can be found on [Gitea Docs site](https://docs.gitea.com/installation/install-with-docker-rootless)
|
Documentation on using docker image can be found on [Gitea Docs site](https://docs.gitea.com/installation/install-with-docker-rootless).
|
||||||
|
7
docs/.gitignore
vendored
7
docs/.gitignore
vendored
@ -1,7 +0,0 @@
|
|||||||
public/
|
|
||||||
templates/swagger/v1_json.tmpl
|
|
||||||
themes/
|
|
||||||
resources/
|
|
||||||
|
|
||||||
# Temporary lock file while building
|
|
||||||
/.hugo_build.lock
|
|
202
docs/LICENSE
202
docs/LICENSE
@ -1,202 +0,0 @@
|
|||||||
|
|
||||||
Apache License
|
|
||||||
Version 2.0, January 2004
|
|
||||||
http://www.apache.org/licenses/
|
|
||||||
|
|
||||||
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
|
||||||
|
|
||||||
1. Definitions.
|
|
||||||
|
|
||||||
"License" shall mean the terms and conditions for use, reproduction,
|
|
||||||
and distribution as defined by Sections 1 through 9 of this document.
|
|
||||||
|
|
||||||
"Licensor" shall mean the copyright owner or entity authorized by
|
|
||||||
the copyright owner that is granting the License.
|
|
||||||
|
|
||||||
"Legal Entity" shall mean the union of the acting entity and all
|
|
||||||
other entities that control, are controlled by, or are under common
|
|
||||||
control with that entity. For the purposes of this definition,
|
|
||||||
"control" means (i) the power, direct or indirect, to cause the
|
|
||||||
direction or management of such entity, whether by contract or
|
|
||||||
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
|
||||||
outstanding shares, or (iii) beneficial ownership of such entity.
|
|
||||||
|
|
||||||
"You" (or "Your") shall mean an individual or Legal Entity
|
|
||||||
exercising permissions granted by this License.
|
|
||||||
|
|
||||||
"Source" form shall mean the preferred form for making modifications,
|
|
||||||
including but not limited to software source code, documentation
|
|
||||||
source, and configuration files.
|
|
||||||
|
|
||||||
"Object" form shall mean any form resulting from mechanical
|
|
||||||
transformation or translation of a Source form, including but
|
|
||||||
not limited to compiled object code, generated documentation,
|
|
||||||
and conversions to other media types.
|
|
||||||
|
|
||||||
"Work" shall mean the work of authorship, whether in Source or
|
|
||||||
Object form, made available under the License, as indicated by a
|
|
||||||
copyright notice that is included in or attached to the work
|
|
||||||
(an example is provided in the Appendix below).
|
|
||||||
|
|
||||||
"Derivative Works" shall mean any work, whether in Source or Object
|
|
||||||
form, that is based on (or derived from) the Work and for which the
|
|
||||||
editorial revisions, annotations, elaborations, or other modifications
|
|
||||||
represent, as a whole, an original work of authorship. For the purposes
|
|
||||||
of this License, Derivative Works shall not include works that remain
|
|
||||||
separable from, or merely link (or bind by name) to the interfaces of,
|
|
||||||
the Work and Derivative Works thereof.
|
|
||||||
|
|
||||||
"Contribution" shall mean any work of authorship, including
|
|
||||||
the original version of the Work and any modifications or additions
|
|
||||||
to that Work or Derivative Works thereof, that is intentionally
|
|
||||||
submitted to Licensor for inclusion in the Work by the copyright owner
|
|
||||||
or by an individual or Legal Entity authorized to submit on behalf of
|
|
||||||
the copyright owner. For the purposes of this definition, "submitted"
|
|
||||||
means any form of electronic, verbal, or written communication sent
|
|
||||||
to the Licensor or its representatives, including but not limited to
|
|
||||||
communication on electronic mailing lists, source code control systems,
|
|
||||||
and issue tracking systems that are managed by, or on behalf of, the
|
|
||||||
Licensor for the purpose of discussing and improving the Work, but
|
|
||||||
excluding communication that is conspicuously marked or otherwise
|
|
||||||
designated in writing by the copyright owner as "Not a Contribution."
|
|
||||||
|
|
||||||
"Contributor" shall mean Licensor and any individual or Legal Entity
|
|
||||||
on behalf of whom a Contribution has been received by Licensor and
|
|
||||||
subsequently incorporated within the Work.
|
|
||||||
|
|
||||||
2. Grant of Copyright License. Subject to the terms and conditions of
|
|
||||||
this License, each Contributor hereby grants to You a perpetual,
|
|
||||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
|
||||||
copyright license to reproduce, prepare Derivative Works of,
|
|
||||||
publicly display, publicly perform, sublicense, and distribute the
|
|
||||||
Work and such Derivative Works in Source or Object form.
|
|
||||||
|
|
||||||
3. Grant of Patent License. Subject to the terms and conditions of
|
|
||||||
this License, each Contributor hereby grants to You a perpetual,
|
|
||||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
|
||||||
(except as stated in this section) patent license to make, have made,
|
|
||||||
use, offer to sell, sell, import, and otherwise transfer the Work,
|
|
||||||
where such license applies only to those patent claims licensable
|
|
||||||
by such Contributor that are necessarily infringed by their
|
|
||||||
Contribution(s) alone or by combination of their Contribution(s)
|
|
||||||
with the Work to which such Contribution(s) was submitted. If You
|
|
||||||
institute patent litigation against any entity (including a
|
|
||||||
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
|
||||||
or a Contribution incorporated within the Work constitutes direct
|
|
||||||
or contributory patent infringement, then any patent licenses
|
|
||||||
granted to You under this License for that Work shall terminate
|
|
||||||
as of the date such litigation is filed.
|
|
||||||
|
|
||||||
4. Redistribution. You may reproduce and distribute copies of the
|
|
||||||
Work or Derivative Works thereof in any medium, with or without
|
|
||||||
modifications, and in Source or Object form, provided that You
|
|
||||||
meet the following conditions:
|
|
||||||
|
|
||||||
(a) You must give any other recipients of the Work or
|
|
||||||
Derivative Works a copy of this License; and
|
|
||||||
|
|
||||||
(b) You must cause any modified files to carry prominent notices
|
|
||||||
stating that You changed the files; and
|
|
||||||
|
|
||||||
(c) You must retain, in the Source form of any Derivative Works
|
|
||||||
that You distribute, all copyright, patent, trademark, and
|
|
||||||
attribution notices from the Source form of the Work,
|
|
||||||
excluding those notices that do not pertain to any part of
|
|
||||||
the Derivative Works; and
|
|
||||||
|
|
||||||
(d) If the Work includes a "NOTICE" text file as part of its
|
|
||||||
distribution, then any Derivative Works that You distribute must
|
|
||||||
include a readable copy of the attribution notices contained
|
|
||||||
within such NOTICE file, excluding those notices that do not
|
|
||||||
pertain to any part of the Derivative Works, in at least one
|
|
||||||
of the following places: within a NOTICE text file distributed
|
|
||||||
as part of the Derivative Works; within the Source form or
|
|
||||||
documentation, if provided along with the Derivative Works; or,
|
|
||||||
within a display generated by the Derivative Works, if and
|
|
||||||
wherever such third-party notices normally appear. The contents
|
|
||||||
of the NOTICE file are for informational purposes only and
|
|
||||||
do not modify the License. You may add Your own attribution
|
|
||||||
notices within Derivative Works that You distribute, alongside
|
|
||||||
or as an addendum to the NOTICE text from the Work, provided
|
|
||||||
that such additional attribution notices cannot be construed
|
|
||||||
as modifying the License.
|
|
||||||
|
|
||||||
You may add Your own copyright statement to Your modifications and
|
|
||||||
may provide additional or different license terms and conditions
|
|
||||||
for use, reproduction, or distribution of Your modifications, or
|
|
||||||
for any such Derivative Works as a whole, provided Your use,
|
|
||||||
reproduction, and distribution of the Work otherwise complies with
|
|
||||||
the conditions stated in this License.
|
|
||||||
|
|
||||||
5. Submission of Contributions. Unless You explicitly state otherwise,
|
|
||||||
any Contribution intentionally submitted for inclusion in the Work
|
|
||||||
by You to the Licensor shall be under the terms and conditions of
|
|
||||||
this License, without any additional terms or conditions.
|
|
||||||
Notwithstanding the above, nothing herein shall supersede or modify
|
|
||||||
the terms of any separate license agreement you may have executed
|
|
||||||
with Licensor regarding such Contributions.
|
|
||||||
|
|
||||||
6. Trademarks. This License does not grant permission to use the trade
|
|
||||||
names, trademarks, service marks, or product names of the Licensor,
|
|
||||||
except as required for reasonable and customary use in describing the
|
|
||||||
origin of the Work and reproducing the content of the NOTICE file.
|
|
||||||
|
|
||||||
7. Disclaimer of Warranty. Unless required by applicable law or
|
|
||||||
agreed to in writing, Licensor provides the Work (and each
|
|
||||||
Contributor provides its Contributions) on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
||||||
implied, including, without limitation, any warranties or conditions
|
|
||||||
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
|
||||||
PARTICULAR PURPOSE. You are solely responsible for determining the
|
|
||||||
appropriateness of using or redistributing the Work and assume any
|
|
||||||
risks associated with Your exercise of permissions under this License.
|
|
||||||
|
|
||||||
8. Limitation of Liability. In no event and under no legal theory,
|
|
||||||
whether in tort (including negligence), contract, or otherwise,
|
|
||||||
unless required by applicable law (such as deliberate and grossly
|
|
||||||
negligent acts) or agreed to in writing, shall any Contributor be
|
|
||||||
liable to You for damages, including any direct, indirect, special,
|
|
||||||
incidental, or consequential damages of any character arising as a
|
|
||||||
result of this License or out of the use or inability to use the
|
|
||||||
Work (including but not limited to damages for loss of goodwill,
|
|
||||||
work stoppage, computer failure or malfunction, or any and all
|
|
||||||
other commercial damages or losses), even if such Contributor
|
|
||||||
has been advised of the possibility of such damages.
|
|
||||||
|
|
||||||
9. Accepting Warranty or Additional Liability. While redistributing
|
|
||||||
the Work or Derivative Works thereof, You may choose to offer,
|
|
||||||
and charge a fee for, acceptance of support, warranty, indemnity,
|
|
||||||
or other liability obligations and/or rights consistent with this
|
|
||||||
License. However, in accepting such obligations, You may act only
|
|
||||||
on Your own behalf and on Your sole responsibility, not on behalf
|
|
||||||
of any other Contributor, and only if You agree to indemnify,
|
|
||||||
defend, and hold each Contributor harmless for any liability
|
|
||||||
incurred by, or claims asserted against, such Contributor by reason
|
|
||||||
of your accepting any such warranty or additional liability.
|
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
APPENDIX: How to apply the Apache License to your work.
|
|
||||||
|
|
||||||
To apply the Apache License to your work, attach the following
|
|
||||||
boilerplate notice, with the fields enclosed by brackets "[]"
|
|
||||||
replaced with your own identifying information. (Don't include
|
|
||||||
the brackets!) The text should be enclosed in the appropriate
|
|
||||||
comment syntax for the file format. We also recommend that a
|
|
||||||
file or class name and description of purpose be included on the
|
|
||||||
same "printed page" as the copyright notice for easier
|
|
||||||
identification within third-party archives.
|
|
||||||
|
|
||||||
Copyright [yyyy] [name of copyright owner]
|
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
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 "AS IS" 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.
|
|
@ -1,22 +0,0 @@
|
|||||||
# Gitea: Docs
|
|
||||||
|
|
||||||
[![Join the chat at https://img.shields.io/discord/322538954119184384.svg](https://img.shields.io/discord/322538954119184384.svg)](https://discord.gg/Gitea)
|
|
||||||
[![](https://images.microbadger.com/badges/image/gitea/docs.svg)](http://microbadger.com/images/gitea/docs "Get your own image badge on microbadger.com")
|
|
||||||
|
|
||||||
These docs are ingested by our [docs repo](https://gitea.com/gitea/gitea-docusaurus).
|
|
||||||
|
|
||||||
## Authors
|
|
||||||
|
|
||||||
* [Maintainers](https://github.com/orgs/go-gitea/people)
|
|
||||||
* [Contributors](https://github.com/go-gitea/docs/graphs/contributors)
|
|
||||||
|
|
||||||
## License
|
|
||||||
|
|
||||||
This project is under the Apache-2.0 License. See the [LICENSE](LICENSE) file
|
|
||||||
for the full license text.
|
|
||||||
|
|
||||||
## Copyright
|
|
||||||
|
|
||||||
```
|
|
||||||
Copyright (c) 2016 The Gitea Authors <https://gitea.io>
|
|
||||||
```
|
|
@ -1,22 +0,0 @@
|
|||||||
# Gitea: 文档
|
|
||||||
|
|
||||||
[![Build Status](http://drone.gitea.io/api/badges/go-gitea/docs/status.svg)](http://drone.gitea.io/go-gitea/docs)
|
|
||||||
[![Join the chat at https://img.shields.io/discord/322538954119184384.svg](https://img.shields.io/discord/322538954119184384.svg)](https://discord.gg/Gitea)
|
|
||||||
[![](https://images.microbadger.com/badges/image/gitea/docs.svg)](http://microbadger.com/images/gitea/docs "Get your own image badge on microbadger.com")
|
|
||||||
|
|
||||||
https://gitea.com/gitea/gitea-docusaurus
|
|
||||||
|
|
||||||
## 关于我们
|
|
||||||
|
|
||||||
* [维护者信息](https://github.com/orgs/go-gitea/people)
|
|
||||||
* [代码贡献者信息](https://github.com/go-gitea/docs/graphs/contributors)
|
|
||||||
|
|
||||||
## 许可证
|
|
||||||
|
|
||||||
此项目采用 Apache-2.0 许可协议,请参见 [协议文件](LICENSE) 获取更多信息。
|
|
||||||
|
|
||||||
## 版权声明
|
|
||||||
|
|
||||||
```
|
|
||||||
Copyright (c) 2016 The Gitea Authors <https://gitea.io>
|
|
||||||
```
|
|
@ -1,13 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2023-04-27T14:00:00+08:00"
|
|
||||||
title: "Actions"
|
|
||||||
slug: "actions"
|
|
||||||
sidebar_position: 36
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
name: "Usage - Actions"
|
|
||||||
sidebar_position: 31
|
|
||||||
identifier: "actions"
|
|
||||||
---
|
|
@ -1,14 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2016-12-01T16:00:00+02:00"
|
|
||||||
title: "Administration"
|
|
||||||
slug: "administration"
|
|
||||||
sidebar_position: 30
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
name: "Administration"
|
|
||||||
sidebar_position: 20
|
|
||||||
collapse: true
|
|
||||||
identifier: "administration"
|
|
||||||
---
|
|
@ -1,13 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2016-12-01T16:00:00+02:00"
|
|
||||||
title: "运维"
|
|
||||||
slug: "administration"
|
|
||||||
sidebar_position: 30
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
name: "运维"
|
|
||||||
sidebar_position: 20
|
|
||||||
identifier: "administration"
|
|
||||||
---
|
|
@ -1,40 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2019-12-28"
|
|
||||||
title: "Adding Legal Pages"
|
|
||||||
slug: adding-legal-pages
|
|
||||||
sidebar_position: 110
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/adding-legal-pages
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Adding Legal Pages"
|
|
||||||
identifier: "adding-legal-pages"
|
|
||||||
sidebar_position: 110
|
|
||||||
---
|
|
||||||
|
|
||||||
Some jurisdictions (such as EU), requires certain legal pages (e.g. Privacy Policy) to be added to website. Follow these steps to add them to your Gitea instance.
|
|
||||||
|
|
||||||
## Getting Pages
|
|
||||||
|
|
||||||
Gitea source code ships with sample pages, available in `contrib/legal` directory. Copy them to `custom/public/assets/`. For example, to add Privacy Policy:
|
|
||||||
|
|
||||||
```
|
|
||||||
wget -O /path/to/custom/public/assets/privacy.html https://raw.githubusercontent.com/go-gitea/gitea/main/contrib/legal/privacy.html.sample
|
|
||||||
```
|
|
||||||
|
|
||||||
Now you need to edit the page to meet your requirements. In particular you must change the email addresses, web addresses and references to "Your Gitea Instance" to match your situation.
|
|
||||||
|
|
||||||
You absolutely must not place a general ToS or privacy statement that implies that the Gitea project is responsible for your server.
|
|
||||||
|
|
||||||
## Make it Visible
|
|
||||||
|
|
||||||
Create or append to `/path/to/custom/templates/custom/extra_links_footer.tmpl`:
|
|
||||||
|
|
||||||
```go
|
|
||||||
<a class="item" href="{{AppSubUrl}}/assets/privacy.html">Privacy Policy</a>
|
|
||||||
```
|
|
||||||
|
|
||||||
Restart Gitea to see the changes.
|
|
@ -1,40 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2023-05-23T09:00:00+08:00"
|
|
||||||
title: "添加法律页面"
|
|
||||||
slug: adding-legal-pages
|
|
||||||
sidebar_position: 110
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/adding-legal-pages
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "添加法律页面"
|
|
||||||
identifier: "adding-legal-pages"
|
|
||||||
sidebar_position: 110
|
|
||||||
---
|
|
||||||
|
|
||||||
一些法域(例如欧盟)要求在网站上添加特定的法律页面(例如隐私政策)。按照以下步骤将它们添加到你的 Gitea 实例中。
|
|
||||||
|
|
||||||
## 获取页面
|
|
||||||
|
|
||||||
Gitea 源代码附带了示例页面,位于 `contrib/legal` 目录中。将它们复制到 `custom/public/assets/` 目录下。例如,如果要添加隐私政策:
|
|
||||||
|
|
||||||
```
|
|
||||||
wget -O /path/to/custom/public/assets/privacy.html https://raw.githubusercontent.com/go-gitea/gitea/main/contrib/legal/privacy.html.sample
|
|
||||||
```
|
|
||||||
|
|
||||||
现在,你需要编辑该页面以满足你的需求。特别是,你必须更改电子邮件地址、网址以及与 "Your Gitea Instance" 相关的引用,以匹配你的情况。
|
|
||||||
|
|
||||||
请务必不要放置会暗示 Gitea 项目对你的服务器负责的一般服务条款或隐私声明。
|
|
||||||
|
|
||||||
## 使其可见
|
|
||||||
|
|
||||||
创建或追加到 `/path/to/custom/templates/custom/extra_links_footer.tmpl` 文件中:
|
|
||||||
|
|
||||||
```go
|
|
||||||
<a class="item" href="{{AppSubUrl}}/assets/privacy.html">隐私政策</a>
|
|
||||||
```
|
|
||||||
|
|
||||||
重启 Gitea 以查看更改。
|
|
@ -1,160 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2017-01-01T16:00:00+02:00"
|
|
||||||
title: "Backup and Restore"
|
|
||||||
slug: "backup-and-restore"
|
|
||||||
sidebar_position: 11
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/backup-and-restore
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Backup and Restore"
|
|
||||||
sidebar_position: 11
|
|
||||||
identifier: "backup-and-restore"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Backup and Restore
|
|
||||||
|
|
||||||
Gitea currently has a `dump` command that will save the installation to a ZIP file. This
|
|
||||||
file can be unpacked and used to restore an instance.
|
|
||||||
|
|
||||||
## Backup Consistency
|
|
||||||
|
|
||||||
To ensure the consistency of the Gitea instance, it must be shutdown during backup.
|
|
||||||
|
|
||||||
Gitea consists of a database, files and git repositories, all of which change when it is used. For instance, when a migration is in progress, a transaction is created in the database while the git repository is being copied over. If the backup happens in the middle of the migration, the git repository may be incomplete although the database claims otherwise because it was dumped afterwards. The only way to avoid such race conditions is by stopping the Gitea instance during the backups.
|
|
||||||
|
|
||||||
## Backup Command (`dump`)
|
|
||||||
|
|
||||||
Switch to the user running Gitea: `su git`. Run `./gitea dump -c /path/to/app.ini` in the Gitea installation
|
|
||||||
directory. There should be some output similar to the following:
|
|
||||||
|
|
||||||
```none
|
|
||||||
2016/12/27 22:32:09 Creating tmp work dir: /tmp/gitea-dump-417443001
|
|
||||||
2016/12/27 22:32:09 Dumping local repositories.../home/git/gitea-repositories
|
|
||||||
2016/12/27 22:32:22 Dumping database...
|
|
||||||
2016/12/27 22:32:22 Packing dump files...
|
|
||||||
2016/12/27 22:32:34 Removing tmp work dir: /tmp/gitea-dump-417443001
|
|
||||||
2016/12/27 22:32:34 Finish dumping in file gitea-dump-1482906742.zip
|
|
||||||
```
|
|
||||||
|
|
||||||
Inside the `gitea-dump-1482906742.zip` file, will be the following:
|
|
||||||
|
|
||||||
- `app.ini` - Optional copy of configuration file if originally stored outside the default `custom/` directory
|
|
||||||
- `custom/` - All config or customization files in `custom/`.
|
|
||||||
- `data/` - Data directory (APP_DATA_PATH), except sessions if you are using file session. This directory includes `attachments`, `avatars`, `lfs`, `indexers`, SQLite file if you are using SQLite.
|
|
||||||
- `repos/` - Complete copy of the repository directory.
|
|
||||||
- `gitea-db.sql` - SQL dump of database
|
|
||||||
- `log/` - Various logs. They are not needed for a recovery or migration.
|
|
||||||
|
|
||||||
Intermediate backup files are created in a temporary directory specified either with the
|
|
||||||
`--tempdir` command-line parameter or the `TMPDIR` environment variable.
|
|
||||||
|
|
||||||
## Backup the database
|
|
||||||
|
|
||||||
The SQL dump created by `gitea dump` uses XORM and Gitea admins may prefer to use the native the MySQL and PostgreSQL dump tools instead. There are still open issues when using XORM for dumping the database that may cause problems when attempting to restore it.
|
|
||||||
|
|
||||||
```sh
|
|
||||||
# mysql
|
|
||||||
mysqldump -u$USER -p$PASS --database $DATABASE > gitea-db.sql
|
|
||||||
# postgres
|
|
||||||
pg_dump -U $USER $DATABASE > gitea-db.sql
|
|
||||||
```
|
|
||||||
|
|
||||||
### Using Docker (`dump`)
|
|
||||||
|
|
||||||
There are a few caveats for using the `dump` command with Docker.
|
|
||||||
|
|
||||||
The command has to be executed with the `RUN_USER = <OS_USERNAME>` specified in `gitea/conf/app.ini`; and, for the zipping of the backup folder to occur without permission error the command `docker exec` must be executed inside of the `--tempdir`.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```none
|
|
||||||
docker exec -u <OS_USERNAME> -it -w <--tempdir> $(docker ps -qf 'name=^<NAME_OF_DOCKER_CONTAINER>$') bash -c '/usr/local/bin/gitea dump -c </path/to/app.ini>'
|
|
||||||
```
|
|
||||||
|
|
||||||
\*Note: `--tempdir` refers to the temporary directory of the docker environment used by Gitea; if you have not specified a custom `--tempdir`, then Gitea uses `/tmp` or the `TMPDIR` environment variable of the docker container. For `--tempdir` adjust your `docker exec` command options accordingly.
|
|
||||||
|
|
||||||
The result should be a file, stored in the `--tempdir` specified, along the lines of: `gitea-dump-1482906742.zip`
|
|
||||||
|
|
||||||
## Restore Command (`restore`)
|
|
||||||
|
|
||||||
There is currently no support for a recovery command. It is a manual process that mostly
|
|
||||||
involves moving files to their correct locations and restoring a database dump.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
unzip gitea-dump-1610949662.zip
|
|
||||||
cd gitea-dump-1610949662
|
|
||||||
mv app.ini /etc/gitea/conf/app.ini
|
|
||||||
mv data/* /var/lib/gitea/data/
|
|
||||||
mv log/* /var/lib/gitea/log/
|
|
||||||
mv repos/* /var/lib/gitea/data/gitea-repositories/
|
|
||||||
chown -R gitea:gitea /etc/gitea/conf/app.ini /var/lib/gitea
|
|
||||||
|
|
||||||
# mysql
|
|
||||||
mysql --default-character-set=utf8mb4 -u$USER -p$PASS $DATABASE <gitea-db.sql
|
|
||||||
# sqlite3
|
|
||||||
sqlite3 $DATABASE_PATH <gitea-db.sql
|
|
||||||
# postgres
|
|
||||||
psql -U $USER -d $DATABASE < gitea-db.sql
|
|
||||||
|
|
||||||
service gitea restart
|
|
||||||
```
|
|
||||||
|
|
||||||
Repository Git Hooks should be regenerated if installation method is changed (eg. binary -> Docker), or if Gitea is installed to a different directory than the previous installation.
|
|
||||||
|
|
||||||
With Gitea running, and from the directory Gitea's binary is located, execute: `./gitea admin regenerate hooks`
|
|
||||||
|
|
||||||
This ensures that application and configuration file paths in repository Git Hooks are consistent and applicable to the current installation. If these paths are not updated, repository `push` actions will fail.
|
|
||||||
|
|
||||||
If you still have issues, consider running `./gitea doctor check` to inspect possible errors (or run with `--fix`).
|
|
||||||
|
|
||||||
### Using Docker (`restore`)
|
|
||||||
|
|
||||||
There is also no support for a recovery command in a Docker-based gitea instance. The restore process contains the same steps as described in the previous section but with different paths.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
# open bash session in container
|
|
||||||
docker exec --user git -it 2a83b293548e bash
|
|
||||||
# unzip your backup file within the container
|
|
||||||
unzip gitea-dump-1610949662.zip
|
|
||||||
cd gitea-dump-1610949662
|
|
||||||
# restore the gitea data
|
|
||||||
mv data/* /data/gitea
|
|
||||||
# restore the repositories itself
|
|
||||||
mv repos/* /data/git/gitea-repositories/
|
|
||||||
# adjust file permissions
|
|
||||||
chown -R git:git /data
|
|
||||||
# Regenerate Git Hooks
|
|
||||||
/usr/local/bin/gitea -c '/data/gitea/conf/app.ini' admin regenerate hooks
|
|
||||||
```
|
|
||||||
|
|
||||||
The default user in the gitea container is `git` (1000:1000). Please replace `2a83b293548e` with your gitea container id or name.
|
|
||||||
|
|
||||||
### Using Docker-rootless (`restore`)
|
|
||||||
|
|
||||||
The restore workflow in Docker-rootless containers differs only in the directories to be used:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
# open bash session in container
|
|
||||||
docker exec --user git -it 2a83b293548e bash
|
|
||||||
# unzip your backup file within the container
|
|
||||||
unzip gitea-dump-1610949662.zip
|
|
||||||
cd gitea-dump-1610949662
|
|
||||||
# restore the app.ini
|
|
||||||
mv data/conf/app.ini /etc/gitea/app.ini
|
|
||||||
# restore the gitea data
|
|
||||||
mv data/* /var/lib/gitea
|
|
||||||
# restore the repositories itself
|
|
||||||
mv repos/* /var/lib/gitea/git/gitea-repositories
|
|
||||||
# adjust file permissions
|
|
||||||
chown -R git:git /etc/gitea/app.ini /var/lib/gitea
|
|
||||||
# Regenerate Git Hooks
|
|
||||||
/usr/local/bin/gitea -c '/etc/gitea/app.ini' admin regenerate hooks
|
|
||||||
```
|
|
@ -1,154 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2018-06-06T09:33:00+08:00"
|
|
||||||
title: "备份与恢复"
|
|
||||||
slug: "backup-and-restore"
|
|
||||||
sidebar_position: 11
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/backup-and-restore
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "备份与恢复"
|
|
||||||
sidebar_position: 11
|
|
||||||
identifier: "backup-and-restore"
|
|
||||||
---
|
|
||||||
|
|
||||||
# 备份与恢复
|
|
||||||
|
|
||||||
Gitea 已经实现了 `dump` 命令可以用来备份所有需要的文件到一个zip压缩文件。该压缩文件可以被用来进行数据恢复。
|
|
||||||
|
|
||||||
## 备份一致性
|
|
||||||
|
|
||||||
为了确保 Gitea 实例的一致性,在备份期间必须关闭它。
|
|
||||||
|
|
||||||
Gitea 包括数据库、文件和 Git 仓库,当它被使用时所有这些都会发生变化。例如,当迁移正在进行时,在数据库中创建一个事务,而 Git 仓库正在被复制。如果备份发生在迁移的中间,Git 仓库可能是不完整的,尽管数据库声称它是完整的,因为它是在之后被转储的。避免这种竞争条件的唯一方法是在备份期间停止 Gitea 实例。
|
|
||||||
|
|
||||||
## 备份命令 (`dump`)
|
|
||||||
|
|
||||||
先转到git用户的权限: `su git`. 再Gitea目录运行 `./gitea dump`。一般会显示类似如下的输出:
|
|
||||||
|
|
||||||
```
|
|
||||||
2016/12/27 22:32:09 Creating tmp work dir: /tmp/gitea-dump-417443001
|
|
||||||
2016/12/27 22:32:09 Dumping local repositories.../home/git/gitea-repositories
|
|
||||||
2016/12/27 22:32:22 Dumping database...
|
|
||||||
2016/12/27 22:32:22 Packing dump files...
|
|
||||||
2016/12/27 22:32:34 Removing tmp work dir: /tmp/gitea-dump-417443001
|
|
||||||
2016/12/27 22:32:34 Finish dumping in file gitea-dump-1482906742.zip
|
|
||||||
```
|
|
||||||
|
|
||||||
最后生成的 `gitea-dump-1482906742.zip` 文件将会包含如下内容:
|
|
||||||
|
|
||||||
* `app.ini` - 如果原先存储在默认的 custom/ 目录之外,则是配置文件的可选副本
|
|
||||||
* `custom/` - 所有保存在 `custom/` 目录下的配置和自定义的文件。
|
|
||||||
* `data/` - 数据目录(APP_DATA_PATH),如果使用文件会话,则不包括会话。该目录包括 `attachments`、`avatars`、`lfs`、`indexers`、如果使用 SQLite 则包括 SQLite 文件。
|
|
||||||
* `repos/` - 仓库目录的完整副本。
|
|
||||||
* `gitea-db.sql` - 数据库dump出来的 SQL。
|
|
||||||
* `log/` - Logs文件,如果用作迁移不是必须的。
|
|
||||||
|
|
||||||
中间备份文件将会在临时目录进行创建,如果您要重新指定临时目录,可以用 `--tempdir` 参数,或者用 `TMPDIR` 环境变量。
|
|
||||||
|
|
||||||
## 备份数据库
|
|
||||||
|
|
||||||
`gitea dump` 创建的 SQL 转储使用 XORM,Gitea 管理员可能更喜欢使用本地的 MySQL 和 PostgreSQL 转储工具。使用 XORM 转储数据库时仍然存在一些问题,可能会导致在尝试恢复时出现问题。
|
|
||||||
|
|
||||||
```sh
|
|
||||||
# mysql
|
|
||||||
mysqldump -u$USER -p$PASS --database $DATABASE > gitea-db.sql
|
|
||||||
# postgres
|
|
||||||
pg_dump -U $USER $DATABASE > gitea-db.sql
|
|
||||||
```
|
|
||||||
|
|
||||||
### 使用Docker (`dump`)
|
|
||||||
|
|
||||||
在使用 Docker 时,使用 `dump` 命令有一些注意事项。
|
|
||||||
|
|
||||||
必须以 `gitea/conf/app.ini` 中指定的 `RUN_USER = <OS_USERNAME>` 执行该命令;并且,为了让备份文件夹的压缩过程能够顺利执行,`docker exec` 命令必须在 `--tempdir` 内部执行。
|
|
||||||
|
|
||||||
示例:
|
|
||||||
|
|
||||||
```none
|
|
||||||
docker exec -u <OS_USERNAME> -it -w <--tempdir> $(docker ps -qf 'name=^<NAME_OF_DOCKER_CONTAINER>$') bash -c '/usr/local/bin/gitea dump -c </path/to/app.ini>'
|
|
||||||
```
|
|
||||||
|
|
||||||
\*注意:`--tempdir` 指的是 Gitea 使用的 Docker 环境的临时目录;如果您没有指定自定义的 `--tempdir`,那么 Gitea 将使用 `/tmp` 或 Docker 容器的 `TMPDIR` 环境变量。对于 `--tempdir`,请相应调整您的 `docker exec` 命令选项。
|
|
||||||
|
|
||||||
结果应该是一个文件,存储在指定的 `--tempdir` 中,类似于:`gitea-dump-1482906742.zip`
|
|
||||||
|
|
||||||
## 恢复命令 (`restore`)
|
|
||||||
|
|
||||||
当前还没有恢复命令,恢复需要人工进行。主要是把文件和数据库进行恢复。
|
|
||||||
|
|
||||||
例如:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
unzip gitea-dump-1610949662.zip
|
|
||||||
cd gitea-dump-1610949662
|
|
||||||
mv app.ini /etc/gitea/conf/app.ini
|
|
||||||
mv data/* /var/lib/gitea/data/
|
|
||||||
mv log/* /var/lib/gitea/log/
|
|
||||||
mv repos/* /var/lib/gitea/gitea-repositories/
|
|
||||||
chown -R gitea:gitea /etc/gitea/conf/app.ini /var/lib/gitea
|
|
||||||
|
|
||||||
# mysql
|
|
||||||
mysql --default-character-set=utf8mb4 -u$USER -p$PASS $DATABASE <gitea-db.sql
|
|
||||||
# sqlite3
|
|
||||||
sqlite3 $DATABASE_PATH <gitea-db.sql
|
|
||||||
# postgres
|
|
||||||
psql -U $USER -d $DATABASE < gitea-db.sql
|
|
||||||
|
|
||||||
service gitea restart
|
|
||||||
```
|
|
||||||
|
|
||||||
如果安装方式发生了变化(例如 二进制 -> Docker),或者 Gitea 安装到了与之前安装不同的目录,则需要重新生成仓库 Git 钩子。
|
|
||||||
|
|
||||||
在 Gitea 运行时,并从 Gitea 二进制文件所在的目录执行:`./gitea admin regenerate hooks`
|
|
||||||
|
|
||||||
这样可以确保仓库 Git 钩子中的应用程序和配置文件路径与当前安装一致。如果这些路径没有更新,仓库的 `push` 操作将失败。
|
|
||||||
|
|
||||||
### 使用 Docker (`restore`)
|
|
||||||
|
|
||||||
在基于 Docker 的 Gitea 实例中,也没有恢复命令的支持。恢复过程与前面描述的步骤相同,但路径不同。
|
|
||||||
|
|
||||||
示例:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
# 在容器中打开 bash 会话
|
|
||||||
docker exec --user git -it 2a83b293548e bash
|
|
||||||
# 在容器内解压您的备份文件
|
|
||||||
unzip gitea-dump-1610949662.zip
|
|
||||||
cd gitea-dump-1610949662
|
|
||||||
# 恢复 Gitea 数据
|
|
||||||
mv data/* /data/gitea
|
|
||||||
# 恢复仓库本身
|
|
||||||
mv repos/* /data/git/gitea-repositories/
|
|
||||||
# 调整文件权限
|
|
||||||
chown -R git:git /data
|
|
||||||
# 重新生成 Git 钩子
|
|
||||||
/usr/local/bin/gitea -c '/data/gitea/conf/app.ini' admin regenerate hooks
|
|
||||||
```
|
|
||||||
|
|
||||||
Gitea 容器中的默认用户是 `git`(1000:1000)。请用您的 Gitea 容器 ID 或名称替换 `2a83b293548e`。
|
|
||||||
|
|
||||||
### 使用 Docker-rootless (`restore`)
|
|
||||||
|
|
||||||
在 Docker-rootless 容器中的恢复工作流程只是要使用的目录不同:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
# 在容器中打开 bash 会话
|
|
||||||
docker exec --user git -it 2a83b293548e bash
|
|
||||||
# 在容器内解压您的备份文件
|
|
||||||
unzip gitea-dump-1610949662.zip
|
|
||||||
cd gitea-dump-1610949662
|
|
||||||
# 恢复 app.ini
|
|
||||||
mv data/conf/app.ini /etc/gitea/app.ini
|
|
||||||
# 恢复 Gitea 数据
|
|
||||||
mv data/* /var/lib/gitea
|
|
||||||
# 恢复仓库本身
|
|
||||||
mv repos/* /var/lib/gitea/git/gitea-repositories
|
|
||||||
# 调整文件权限
|
|
||||||
chown -R git:git /etc/gitea/app.ini /var/lib/gitea
|
|
||||||
# 重新生成 Git 钩子
|
|
||||||
/usr/local/bin/gitea -c '/etc/gitea/app.ini' admin regenerate hooks
|
|
||||||
```
|
|
@ -1,119 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2020-01-25T21:00:00-03:00"
|
|
||||||
title: "Embedded data extraction tool"
|
|
||||||
slug: "cmd-embedded"
|
|
||||||
sidebar_position: 20
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/cmd-embedded
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Embedded data extraction tool"
|
|
||||||
sidebar_position: 20
|
|
||||||
identifier: "cmd-embedded"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Embedded data extraction tool
|
|
||||||
|
|
||||||
Gitea's executable contains all the resources required to run: templates, images, style-sheets
|
|
||||||
and translations. Any of them can be overridden by placing a replacement in a matching path
|
|
||||||
inside the `custom` directory (see [Customizing Gitea](administration/customizing-gitea.md)).
|
|
||||||
|
|
||||||
To obtain a copy of the embedded resources ready for editing, the `embedded` command from the CLI
|
|
||||||
can be used from the OS shell interface.
|
|
||||||
|
|
||||||
**NOTE:** The embedded data extraction tool is included in Gitea versions 1.12 and above.
|
|
||||||
|
|
||||||
## Listing resources
|
|
||||||
|
|
||||||
To list resources embedded in Gitea's executable, use the following syntax:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
gitea embedded list [--include-vendored] [patterns...]
|
|
||||||
```
|
|
||||||
|
|
||||||
The `--include-vendored` flag makes the command include vendored files, which are
|
|
||||||
normally excluded; that is, files from external libraries that are required for Gitea
|
|
||||||
(e.g. [octicons](https://octicons.github.com/), etc).
|
|
||||||
|
|
||||||
A list of file search patterns can be provided. Gitea uses [gobwas/glob](https://github.com/gobwas/glob)
|
|
||||||
for its glob syntax. Here are some examples:
|
|
||||||
|
|
||||||
- List all template files, in any virtual directory: `**.tmpl`
|
|
||||||
- List all mail template files: `templates/mail/**.tmpl`
|
|
||||||
- List all files inside `public/assets/img`: `public/assets/img/**`
|
|
||||||
|
|
||||||
Don't forget to use quotes for the patterns, as spaces, `*` and other characters might have
|
|
||||||
a special meaning for your command shell.
|
|
||||||
|
|
||||||
If no pattern is provided, all files are listed.
|
|
||||||
|
|
||||||
### Example: Listing all embedded files
|
|
||||||
|
|
||||||
Listing all embedded files with `openid` in their path:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
$ gitea embedded list '**openid**'
|
|
||||||
public/assets/img/auth/openid_connect.svg
|
|
||||||
public/assets/img/openid-16x16.png
|
|
||||||
templates/user/auth/finalize_openid.tmpl
|
|
||||||
templates/user/auth/signin_openid.tmpl
|
|
||||||
templates/user/auth/signup_openid_connect.tmpl
|
|
||||||
templates/user/auth/signup_openid_navbar.tmpl
|
|
||||||
templates/user/auth/signup_openid_register.tmpl
|
|
||||||
templates/user/settings/security_openid.tmpl
|
|
||||||
```
|
|
||||||
|
|
||||||
## Extracting resources
|
|
||||||
|
|
||||||
To extract resources embedded in Gitea's executable, use the following syntax:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
gitea [--config {file}] embedded extract [--destination {dir}|--custom] [--overwrite|--rename] [--include-vendored] {patterns...}
|
|
||||||
```
|
|
||||||
|
|
||||||
The `--config` option tells Gitea the location of the `app.ini` configuration file if
|
|
||||||
it's not in its default location. This option is only used with the `--custom` flag.
|
|
||||||
|
|
||||||
The `--destination` option tells Gitea the directory where the files must be extracted to.
|
|
||||||
The default is the current directory.
|
|
||||||
|
|
||||||
The `--custom` flag tells Gitea to extract the files directly into the `custom` directory.
|
|
||||||
For this to work, the command needs to know the location of the `app.ini` configuration
|
|
||||||
file (`--config`) and, depending of the configuration, be ran from the directory where
|
|
||||||
Gitea normally starts. See [Customizing Gitea](administration/customizing-gitea.md) for details.
|
|
||||||
|
|
||||||
The `--overwrite` flag allows any existing files in the destination directory to be overwritten.
|
|
||||||
|
|
||||||
The `--rename` flag tells Gitea to rename any existing files in the destination directory
|
|
||||||
as `filename.bak`. Previous `.bak` files are overwritten.
|
|
||||||
|
|
||||||
At least one file search pattern must be provided; see `list` subcomand above for pattern
|
|
||||||
syntax and examples.
|
|
||||||
|
|
||||||
### Important notice
|
|
||||||
|
|
||||||
Make sure to **only extract those files that require customization**. Files that
|
|
||||||
are present in the `custom` directory are not upgraded by Gitea's upgrade process.
|
|
||||||
When Gitea is upgraded to a new version (by replacing the executable), many of the
|
|
||||||
embedded files will suffer changes. Gitea will honor and use any files found
|
|
||||||
in the `custom` directory, even if they are old and incompatible.
|
|
||||||
|
|
||||||
### Example: Extracting mail templates
|
|
||||||
|
|
||||||
Extracting mail templates to a temporary directory:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
$ mkdir tempdir
|
|
||||||
$ gitea embedded extract --destination tempdir 'templates/mail/**.tmpl'
|
|
||||||
Extracting to tempdir:
|
|
||||||
tempdir/templates/mail/auth/activate.tmpl
|
|
||||||
tempdir/templates/mail/auth/activate_email.tmpl
|
|
||||||
tempdir/templates/mail/auth/register_notify.tmpl
|
|
||||||
tempdir/templates/mail/auth/reset_passwd.tmpl
|
|
||||||
tempdir/templates/mail/issue/assigned.tmpl
|
|
||||||
tempdir/templates/mail/issue/default.tmpl
|
|
||||||
tempdir/templates/mail/notify/collaborator.tmpl
|
|
||||||
```
|
|
@ -1,101 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2023-05-23T09:00:00+08:00"
|
|
||||||
title: "嵌入资源提取工具"
|
|
||||||
slug: "cmd-embedded"
|
|
||||||
sidebar_position: 20
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/cmd-embedded
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "嵌入资源提取工具"
|
|
||||||
sidebar_position: 20
|
|
||||||
identifier: "cmd-embedded"
|
|
||||||
---
|
|
||||||
|
|
||||||
# 嵌入资源提取工具
|
|
||||||
|
|
||||||
Gitea 的可执行文件包含了运行所需的所有资源:模板、图片、样式表和翻译文件。你可以通过在 `custom` 目录下的相应路径中放置替换文件来覆盖其中的任何资源(详见 [自定义 Gitea 配置](administration/customizing-gitea.md))。
|
|
||||||
|
|
||||||
要获取嵌入资源的副本以进行编辑,可以使用 CLI 中的 `embedded` 命令,通过操作系统的 shell 执行。
|
|
||||||
|
|
||||||
**注意:** 嵌入资源提取工具包含在 Gitea 1.12 及以上版本中。
|
|
||||||
|
|
||||||
## 资源列表
|
|
||||||
|
|
||||||
要列出嵌入在 Gitea 可执行文件中的资源,请使用以下语法:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
gitea embedded list [--include-vendored] [patterns...]
|
|
||||||
```
|
|
||||||
|
|
||||||
`--include-vendored` 标志使命令包括被供应的文件,这些文件通常被排除在外;即来自外部库的文件,这些文件是 Gitea 所需的(例如 [octicons](https://octicons.github.com/) 等)。
|
|
||||||
|
|
||||||
可以提供一系列文件搜索模式。Gitea 使用 [gobwas/glob](https://github.com/gobwas/glob) 作为其 glob 语法。以下是一些示例:
|
|
||||||
|
|
||||||
- 列出所有模板文件,无论在哪个虚拟目录下:`**.tmpl`
|
|
||||||
- 列出所有邮件模板文件:`templates/mail/**.tmpl`
|
|
||||||
列出 `public/assets/img` 目录下的所有文件:`public/assets/img/**`
|
|
||||||
|
|
||||||
不要忘记为模式使用引号,因为空格、`*` 和其他字符可能对命令行解释器有特殊含义。
|
|
||||||
|
|
||||||
如果未提供模式,则列出所有文件。
|
|
||||||
|
|
||||||
### 示例:列出所有嵌入文件
|
|
||||||
|
|
||||||
列出所有路径中包含 `openid` 的嵌入文件:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
$ gitea embedded list '**openid**'
|
|
||||||
public/assets/img/auth/openid_connect.svg
|
|
||||||
public/assets/img/openid-16x16.png
|
|
||||||
templates/user/auth/finalize_openid.tmpl
|
|
||||||
templates/user/auth/signin_openid.tmpl
|
|
||||||
templates/user/auth/signup_openid_connect.tmpl
|
|
||||||
templates/user/auth/signup_openid_navbar.tmpl
|
|
||||||
templates/user/auth/signup_openid_register.tmpl
|
|
||||||
templates/user/settings/security_openid.tmpl
|
|
||||||
```
|
|
||||||
|
|
||||||
## 提取资源
|
|
||||||
|
|
||||||
要提取嵌入在 Gitea 可执行文件中的资源,请使用以下语法:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
gitea [--config {file}] embedded extract [--destination {dir}|--custom] [--overwrite|--rename] [--include-vendored] {patterns...}
|
|
||||||
```
|
|
||||||
|
|
||||||
`--config` 选项用于告知 Gitea `app.ini` 配置文件的位置(如果不在默认位置)。此选项仅在使用 `--custom` 标志时使用。
|
|
||||||
|
|
||||||
`--destination` 选项用于指定提取文件的目标目录。默认为当前目录。
|
|
||||||
|
|
||||||
`--custom` 标志告知 Gitea 直接将文件提取到 `custom` 目录中。为使其正常工作,该命令需要知道 `app.ini` 配置文件的位置(通过 `--config` 指定),并且根据配置的不同,需要从 Gitea 通常启动的目录运行。有关详细信息,请参阅 [自定义 Gitea 配置](administration/customizing-gitea.md)。
|
|
||||||
|
|
||||||
`--overwrite` 标志允许覆盖目标目录中的任何现有文件。
|
|
||||||
|
|
||||||
`--rename` 标志告知 Gitea 将目标目录中的任何现有文件重命名为 `filename.bak`。之前的 `.bak` 文件将被覆盖。
|
|
||||||
|
|
||||||
至少需要提供一个文件搜索模式;有关模式的语法和示例,请参阅上述 `list` 子命令。
|
|
||||||
|
|
||||||
### 重要提示
|
|
||||||
|
|
||||||
请确保**只提取需要自定义的文件**。位于 `custom` 目录中的文件不会受到 Gitea 的升级过程的影响。当 Gitea 升级到新版本(通过替换可执行文件)时,许多嵌入文件将发生变化。Gitea 将尊重并使用在 `custom` 目录中找到的任何文件,即使这些文件是旧的和不兼容的。
|
|
||||||
|
|
||||||
### 示例:提取邮件模板
|
|
||||||
|
|
||||||
将邮件模板提取到临时目录:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
$ mkdir tempdir
|
|
||||||
$ gitea embedded extract --destination tempdir 'templates/mail/**.tmpl'
|
|
||||||
Extracting to tempdir:
|
|
||||||
tempdir/templates/mail/auth/activate.tmpl
|
|
||||||
tempdir/templates/mail/auth/activate_email.tmpl
|
|
||||||
tempdir/templates/mail/auth/register_notify.tmpl
|
|
||||||
tempdir/templates/mail/auth/reset_passwd.tmpl
|
|
||||||
tempdir/templates/mail/issue/assigned.tmpl
|
|
||||||
tempdir/templates/mail/issue/default.tmpl
|
|
||||||
tempdir/templates/mail/notify/collaborator.tmpl
|
|
||||||
```
|
|
@ -1,572 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2017-01-01T16:00:00+02:00"
|
|
||||||
title: "Gitea Command Line"
|
|
||||||
slug: "command-line"
|
|
||||||
sidebar_position: 1
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/command-line
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Command Line"
|
|
||||||
sidebar_position: 1
|
|
||||||
identifier: "command-line"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Command Line
|
|
||||||
|
|
||||||
## Usage
|
|
||||||
|
|
||||||
`gitea [global options] command [command or global options] [arguments...]`
|
|
||||||
|
|
||||||
## Global options
|
|
||||||
|
|
||||||
All global options can be placed at the command level.
|
|
||||||
|
|
||||||
- `--help`, `-h`: Show help text and exit. Optional.
|
|
||||||
- `--version`, `-v`: Show version and exit. Optional. (example: `Gitea version 1.1.0+218-g7b907ed built with: bindata, sqlite`).
|
|
||||||
- `--work-path path`, `-w path`: Gitea's work path. Optional. (default: the binary's path or `$GITEA_WORK_DIR`)
|
|
||||||
- `--custom-path path`, `-C path`: Gitea's custom folder path. Optional. (default: `WorkPath`/custom or `$GITEA_CUSTOM`).
|
|
||||||
- `--config path`, `-c path`: Gitea configuration file path. Optional. (default: `CustomPath`/conf/app.ini).
|
|
||||||
|
|
||||||
NB: The defaults custom-path, config and work-path can also be
|
|
||||||
changed at build time (if preferred).
|
|
||||||
|
|
||||||
## Commands
|
|
||||||
|
|
||||||
### web
|
|
||||||
|
|
||||||
Starts the server:
|
|
||||||
|
|
||||||
- Options:
|
|
||||||
- `--port number`, `-p number`: Port number. Optional. (default: 3000). Overrides configuration file.
|
|
||||||
- `--install-port number`: Port number to run the install page on. Optional. (default: 3000). Overrides configuration file.
|
|
||||||
- `--pid path`, `-P path`: Pidfile path. Optional.
|
|
||||||
- `--quiet`, `-q`: Only emit Fatal logs on the console for logs emitted before logging set up.
|
|
||||||
- `--verbose`: Emit tracing logs on the console for logs emitted before logging is set-up.
|
|
||||||
- Examples:
|
|
||||||
- `gitea web`
|
|
||||||
- `gitea web --port 80`
|
|
||||||
- `gitea web --config /etc/gitea.ini --pid /some/custom/gitea.pid`
|
|
||||||
- Notes:
|
|
||||||
- Gitea should not be run as root. To bind to a port below 1024, you can use setcap on
|
|
||||||
Linux: `sudo setcap 'cap_net_bind_service=+ep' /path/to/gitea`. This will need to be
|
|
||||||
redone every time you update Gitea.
|
|
||||||
|
|
||||||
### admin
|
|
||||||
|
|
||||||
Admin operations:
|
|
||||||
|
|
||||||
- Commands:
|
|
||||||
- `user`:
|
|
||||||
- `list`:
|
|
||||||
- Options:
|
|
||||||
- `--admin`: List only admin users. Optional.
|
|
||||||
- Description: lists all users that exist
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin user list`
|
|
||||||
- `delete`:
|
|
||||||
- Options:
|
|
||||||
- `--email`: Email of the user to be deleted.
|
|
||||||
- `--username`: Username of user to be deleted.
|
|
||||||
- `--id`: ID of user to be deleted.
|
|
||||||
- One of `--id`, `--username` or `--email` is required. If more than one is provided then all have to match.
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin user delete --id 1`
|
|
||||||
- `create`:
|
|
||||||
- Options:
|
|
||||||
- `--name value`: Username. Required. As of Gitea 1.9.0, use the `--username` flag instead.
|
|
||||||
- `--username value`: Username. Required. New in Gitea 1.9.0.
|
|
||||||
- `--password value`: Password. Required.
|
|
||||||
- `--email value`: Email. Required.
|
|
||||||
- `--admin`: If provided, this makes the user an admin. Optional.
|
|
||||||
- `--access-token`: If provided, an access token will be created for the user. Optional. (default: false).
|
|
||||||
- `--must-change-password`: If provided, the created user will be required to choose a newer password after the
|
|
||||||
initial login. Optional. (default: true).
|
|
||||||
- `--random-password`: If provided, a randomly generated password will be used as the password of the created
|
|
||||||
user. The value of `--password` will be discarded. Optional.
|
|
||||||
- `--random-password-length`: If provided, it will be used to configure the length of the randomly generated
|
|
||||||
password. Optional. (default: 12)
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin user create --username myname --password asecurepassword --email me@example.com`
|
|
||||||
- `change-password`:
|
|
||||||
- Options:
|
|
||||||
- `--username value`, `-u value`: Username. Required.
|
|
||||||
- `--password value`, `-p value`: New password. Required.
|
|
||||||
- `--must-change-password`: If provided, the user is required to choose a new password after the login. Optional.
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin user change-password --username myname --password asecurepassword`
|
|
||||||
- `must-change-password`:
|
|
||||||
- Args:
|
|
||||||
- `[username...]`: Users that must change their passwords
|
|
||||||
- Options:
|
|
||||||
- `--all`, `-A`: Force a password change for all users
|
|
||||||
- `--exclude username`, `-e username`: Exclude the given user. Can be set multiple times.
|
|
||||||
- `--unset`: Revoke forced password change for the given users
|
|
||||||
- `generate-access-token`:
|
|
||||||
- Options:
|
|
||||||
- `--username value`, `-u value`: Username. Required.
|
|
||||||
- `--token-name value`, `-t value`: Token name. Required.
|
|
||||||
- `--scopes value`: Comma-separated list of scopes. Scopes follow the format `[read|write]:<block>` or `all` where `<block>` is one of the available visual groups you can see when opening the API page showing the available routes (for example `repo`).
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin user generate-access-token --username myname --token-name mytoken`
|
|
||||||
- `gitea admin user generate-access-token --help`
|
|
||||||
- `regenerate`
|
|
||||||
- Options:
|
|
||||||
- `hooks`: Regenerate Git Hooks for all repositories
|
|
||||||
- `keys`: Regenerate authorized_keys file
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin regenerate hooks`
|
|
||||||
- `gitea admin regenerate keys`
|
|
||||||
- `auth`:
|
|
||||||
- `list`:
|
|
||||||
- Description: lists all external authentication sources that exist
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin auth list`
|
|
||||||
- `delete`:
|
|
||||||
- Options:
|
|
||||||
- `--id`: ID of source to be deleted. Required.
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin auth delete --id 1`
|
|
||||||
- `add-oauth`:
|
|
||||||
- Options:
|
|
||||||
- `--name`: Application Name.
|
|
||||||
- `--provider`: OAuth2 Provider.
|
|
||||||
- `--key`: Client ID (Key).
|
|
||||||
- `--secret`: Client Secret.
|
|
||||||
- `--auto-discover-url`: OpenID Connect Auto Discovery URL (only required when using OpenID Connect as provider).
|
|
||||||
- `--use-custom-urls`: Use custom URLs for GitLab/GitHub OAuth endpoints.
|
|
||||||
- `--custom-tenant-id`: Use custom Tenant ID for OAuth endpoints.
|
|
||||||
- `--custom-auth-url`: Use a custom Authorization URL (option for GitLab/GitHub).
|
|
||||||
- `--custom-token-url`: Use a custom Token URL (option for GitLab/GitHub).
|
|
||||||
- `--custom-profile-url`: Use a custom Profile URL (option for GitLab/GitHub).
|
|
||||||
- `--custom-email-url`: Use a custom Email URL (option for GitHub).
|
|
||||||
- `--icon-url`: Custom icon URL for OAuth2 login source.
|
|
||||||
- `--skip-local-2fa`: Allow source to override local 2FA. (Optional)
|
|
||||||
- `--scopes`: Additional scopes to request for this OAuth2 source. (Optional)
|
|
||||||
- `--required-claim-name`: Claim name that has to be set to allow users to login with this source. (Optional)
|
|
||||||
- `--required-claim-value`: Claim value that has to be set to allow users to login with this source. (Optional)
|
|
||||||
- `--group-claim-name`: Claim name providing group names for this source. (Optional)
|
|
||||||
- `--admin-group`: Group Claim value for administrator users. (Optional)
|
|
||||||
- `--restricted-group`: Group Claim value for restricted users. (Optional)
|
|
||||||
- `--group-team-map`: JSON mapping between groups and org teams. (Optional)
|
|
||||||
- `--group-team-map-removal`: Activate automatic team membership removal depending on groups. (Optional)
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin auth add-oauth --name external-github --provider github --key OBTAIN_FROM_SOURCE --secret OBTAIN_FROM_SOURCE`
|
|
||||||
- `update-oauth`:
|
|
||||||
- Options:
|
|
||||||
- `--id`: ID of source to be updated. Required.
|
|
||||||
- `--name`: Application Name.
|
|
||||||
- `--provider`: OAuth2 Provider.
|
|
||||||
- `--key`: Client ID (Key).
|
|
||||||
- `--secret`: Client Secret.
|
|
||||||
- `--auto-discover-url`: OpenID Connect Auto Discovery URL (only required when using OpenID Connect as provider).
|
|
||||||
- `--use-custom-urls`: Use custom URLs for GitLab/GitHub OAuth endpoints.
|
|
||||||
- `--custom-tenant-id`: Use custom Tenant ID for OAuth endpoints.
|
|
||||||
- `--custom-auth-url`: Use a custom Authorization URL (option for GitLab/GitHub).
|
|
||||||
- `--custom-token-url`: Use a custom Token URL (option for GitLab/GitHub).
|
|
||||||
- `--custom-profile-url`: Use a custom Profile URL (option for GitLab/GitHub).
|
|
||||||
- `--custom-email-url`: Use a custom Email URL (option for GitHub).
|
|
||||||
- `--icon-url`: Custom icon URL for OAuth2 login source.
|
|
||||||
- `--skip-local-2fa`: Allow source to override local 2FA. (Optional)
|
|
||||||
- `--scopes`: Additional scopes to request for this OAuth2 source.
|
|
||||||
- `--required-claim-name`: Claim name that has to be set to allow users to login with this source. (Optional)
|
|
||||||
- `--required-claim-value`: Claim value that has to be set to allow users to login with this source. (Optional)
|
|
||||||
- `--group-claim-name`: Claim name providing group names for this source. (Optional)
|
|
||||||
- `--admin-group`: Group Claim value for administrator users. (Optional)
|
|
||||||
- `--restricted-group`: Group Claim value for restricted users. (Optional)
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin auth update-oauth --id 1 --name external-github-updated`
|
|
||||||
- `add-smtp`:
|
|
||||||
- Options:
|
|
||||||
- `--name`: Application Name. Required.
|
|
||||||
- `--auth-type`: SMTP Authentication Type (PLAIN/LOGIN/CRAM-MD5). Default to PLAIN.
|
|
||||||
- `--host`: SMTP host. Required.
|
|
||||||
- `--port`: SMTP port. Required.
|
|
||||||
- `--force-smtps`: SMTPS is always used on port 465. Set this to force SMTPS on other ports.
|
|
||||||
- `--skip-verify`: Skip TLS verify.
|
|
||||||
- `--helo-hostname`: Hostname sent with HELO. Leave blank to send current hostname.
|
|
||||||
- `--disable-helo`: Disable SMTP helo.
|
|
||||||
- `--allowed-domains`: Leave empty to allow all domains. Separate multiple domains with a comma (',').
|
|
||||||
- `--skip-local-2fa`: Skip 2FA to log on.
|
|
||||||
- `--active`: This Authentication Source is Activated.
|
|
||||||
Remarks:
|
|
||||||
`--force-smtps`, `--skip-verify`, `--disable-helo`, `--skip-loca-2fs` and `--active` options can be used in form:
|
|
||||||
- `--option`, `--option=true` to enable
|
|
||||||
- `--option=false` to disable
|
|
||||||
If those options are not specified value would not be changed in `update-smtp` or would use default `false` value in `add-smtp`
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin auth add-smtp --name ldap --host smtp.mydomain.org --port 587 --skip-verify --active`
|
|
||||||
- `update-smtp`:
|
|
||||||
- Options:
|
|
||||||
- `--id`: ID of source to be updated. Required.
|
|
||||||
- other options are shared with `add-smtp`
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin auth update-smtp --id 1 --host smtp.mydomain.org --port 587 --skip-verify=false`
|
|
||||||
- `gitea admin auth update-smtp --id 1 --active=false`
|
|
||||||
- `add-ldap`: Add new LDAP (via Bind DN) authentication source
|
|
||||||
- Options:
|
|
||||||
- `--name value`: Authentication name. Required.
|
|
||||||
- `--not-active`: Deactivate the authentication source.
|
|
||||||
- `--security-protocol value`: Security protocol name. Required.
|
|
||||||
- `--skip-tls-verify`: Disable TLS verification.
|
|
||||||
- `--host value`: The address where the LDAP server can be reached. Required.
|
|
||||||
- `--port value`: The port to use when connecting to the LDAP server. Required.
|
|
||||||
- `--user-search-base value`: The LDAP base at which user accounts will be searched for. Required.
|
|
||||||
- `--user-filter value`: An LDAP filter declaring how to find the user record that is attempting to authenticate. Required.
|
|
||||||
- `--admin-filter value`: An LDAP filter specifying if a user should be given administrator privileges.
|
|
||||||
- `--restricted-filter value`: An LDAP filter specifying if a user should be given restricted status.
|
|
||||||
- `--username-attribute value`: The attribute of the user’s LDAP record containing the user name.
|
|
||||||
- `--firstname-attribute value`: The attribute of the user’s LDAP record containing the user’s first name.
|
|
||||||
- `--surname-attribute value`: The attribute of the user’s LDAP record containing the user’s surname.
|
|
||||||
- `--email-attribute value`: The attribute of the user’s LDAP record containing the user’s email address. Required.
|
|
||||||
- `--public-ssh-key-attribute value`: The attribute of the user’s LDAP record containing the user’s public ssh key.
|
|
||||||
- `--avatar-attribute value`: The attribute of the user’s LDAP record containing the user’s avatar.
|
|
||||||
- `--bind-dn value`: The DN to bind to the LDAP server with when searching for the user.
|
|
||||||
- `--bind-password value`: The password for the Bind DN, if any.
|
|
||||||
- `--attributes-in-bind`: Fetch attributes in bind DN context.
|
|
||||||
- `--synchronize-users`: Enable user synchronization.
|
|
||||||
- `--page-size value`: Search page size.
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin auth add-ldap --name ldap --security-protocol unencrypted --host mydomain.org --port 389 --user-search-base "ou=Users,dc=mydomain,dc=org" --user-filter "(&(objectClass=posixAccount)(|(uid=%[1]s)(mail=%[1]s)))" --email-attribute mail`
|
|
||||||
- `update-ldap`: Update existing LDAP (via Bind DN) authentication source
|
|
||||||
- Options:
|
|
||||||
- `--id value`: ID of authentication source. Required.
|
|
||||||
- `--name value`: Authentication name.
|
|
||||||
- `--not-active`: Deactivate the authentication source.
|
|
||||||
- `--security-protocol value`: Security protocol name.
|
|
||||||
- `--skip-tls-verify`: Disable TLS verification.
|
|
||||||
- `--host value`: The address where the LDAP server can be reached.
|
|
||||||
- `--port value`: The port to use when connecting to the LDAP server.
|
|
||||||
- `--user-search-base value`: The LDAP base at which user accounts will be searched for.
|
|
||||||
- `--user-filter value`: An LDAP filter declaring how to find the user record that is attempting to authenticate.
|
|
||||||
- `--admin-filter value`: An LDAP filter specifying if a user should be given administrator privileges.
|
|
||||||
- `--restricted-filter value`: An LDAP filter specifying if a user should be given restricted status.
|
|
||||||
- `--username-attribute value`: The attribute of the user’s LDAP record containing the user name.
|
|
||||||
- `--firstname-attribute value`: The attribute of the user’s LDAP record containing the user’s first name.
|
|
||||||
- `--surname-attribute value`: The attribute of the user’s LDAP record containing the user’s surname.
|
|
||||||
- `--email-attribute value`: The attribute of the user’s LDAP record containing the user’s email address.
|
|
||||||
- `--public-ssh-key-attribute value`: The attribute of the user’s LDAP record containing the user’s public ssh key.
|
|
||||||
- `--avatar-attribute value`: The attribute of the user’s LDAP record containing the user’s avatar.
|
|
||||||
- `--bind-dn value`: The DN to bind to the LDAP server with when searching for the user.
|
|
||||||
- `--bind-password value`: The password for the Bind DN, if any.
|
|
||||||
- `--attributes-in-bind`: Fetch attributes in bind DN context.
|
|
||||||
- `--synchronize-users`: Enable user synchronization.
|
|
||||||
- `--page-size value`: Search page size.
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin auth update-ldap --id 1 --name "my ldap auth source"`
|
|
||||||
- `gitea admin auth update-ldap --id 1 --username-attribute uid --firstname-attribute givenName --surname-attribute sn`
|
|
||||||
- `add-ldap-simple`: Add new LDAP (simple auth) authentication source
|
|
||||||
- Options:
|
|
||||||
- `--name value`: Authentication name. Required.
|
|
||||||
- `--not-active`: Deactivate the authentication source.
|
|
||||||
- `--security-protocol value`: Security protocol name. Required.
|
|
||||||
- `--skip-tls-verify`: Disable TLS verification.
|
|
||||||
- `--host value`: The address where the LDAP server can be reached. Required.
|
|
||||||
- `--port value`: The port to use when connecting to the LDAP server. Required.
|
|
||||||
- `--user-search-base value`: The LDAP base at which user accounts will be searched for.
|
|
||||||
- `--user-filter value`: An LDAP filter declaring how to find the user record that is attempting to authenticate. Required.
|
|
||||||
- `--admin-filter value`: An LDAP filter specifying if a user should be given administrator privileges.
|
|
||||||
- `--restricted-filter value`: An LDAP filter specifying if a user should be given restricted status.
|
|
||||||
- `--username-attribute value`: The attribute of the user’s LDAP record containing the user name.
|
|
||||||
- `--firstname-attribute value`: The attribute of the user’s LDAP record containing the user’s first name.
|
|
||||||
- `--surname-attribute value`: The attribute of the user’s LDAP record containing the user’s surname.
|
|
||||||
- `--email-attribute value`: The attribute of the user’s LDAP record containing the user’s email address. Required.
|
|
||||||
- `--public-ssh-key-attribute value`: The attribute of the user’s LDAP record containing the user’s public ssh key.
|
|
||||||
- `--avatar-attribute value`: The attribute of the user’s LDAP record containing the user’s avatar.
|
|
||||||
- `--user-dn value`: The user’s DN. Required.
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin auth add-ldap-simple --name ldap --security-protocol unencrypted --host mydomain.org --port 389 --user-dn "cn=%s,ou=Users,dc=mydomain,dc=org" --user-filter "(&(objectClass=posixAccount)(cn=%s))" --email-attribute mail`
|
|
||||||
- `update-ldap-simple`: Update existing LDAP (simple auth) authentication source
|
|
||||||
- Options:
|
|
||||||
- `--id value`: ID of authentication source. Required.
|
|
||||||
- `--name value`: Authentication name.
|
|
||||||
- `--not-active`: Deactivate the authentication source.
|
|
||||||
- `--security-protocol value`: Security protocol name.
|
|
||||||
- `--skip-tls-verify`: Disable TLS verification.
|
|
||||||
- `--host value`: The address where the LDAP server can be reached.
|
|
||||||
- `--port value`: The port to use when connecting to the LDAP server.
|
|
||||||
- `--user-search-base value`: The LDAP base at which user accounts will be searched for.
|
|
||||||
- `--user-filter value`: An LDAP filter declaring how to find the user record that is attempting to authenticate.
|
|
||||||
- `--admin-filter value`: An LDAP filter specifying if a user should be given administrator privileges.
|
|
||||||
- `--restricted-filter value`: An LDAP filter specifying if a user should be given restricted status.
|
|
||||||
- `--username-attribute value`: The attribute of the user’s LDAP record containing the user name.
|
|
||||||
- `--firstname-attribute value`: The attribute of the user’s LDAP record containing the user’s first name.
|
|
||||||
- `--surname-attribute value`: The attribute of the user’s LDAP record containing the user’s surname.
|
|
||||||
- `--email-attribute value`: The attribute of the user’s LDAP record containing the user’s email address.
|
|
||||||
- `--public-ssh-key-attribute value`: The attribute of the user’s LDAP record containing the user’s public ssh key.
|
|
||||||
- `--avatar-attribute value`: The attribute of the user’s LDAP record containing the user’s avatar.
|
|
||||||
- `--user-dn value`: The user’s DN.
|
|
||||||
- Examples:
|
|
||||||
- `gitea admin auth update-ldap-simple --id 1 --name "my ldap auth source"`
|
|
||||||
- `gitea admin auth update-ldap-simple --id 1 --username-attribute uid --firstname-attribute givenName --surname-attribute sn`
|
|
||||||
|
|
||||||
### cert
|
|
||||||
|
|
||||||
Generates a self-signed SSL certificate. Outputs to `cert.pem` and `key.pem` in the current
|
|
||||||
directory and will overwrite any existing files.
|
|
||||||
|
|
||||||
- Options:
|
|
||||||
- `--host value`: Comma separated hostnames and ips which this certificate is valid for.
|
|
||||||
Wildcards are supported. Required.
|
|
||||||
- `--ecdsa-curve value`: ECDSA curve to use to generate a key. Optional. Valid options
|
|
||||||
are P224, P256, P384, P521.
|
|
||||||
- `--rsa-bits value`: Size of RSA key to generate. Optional. Ignored if --ecdsa-curve is
|
|
||||||
set. (default: 3072).
|
|
||||||
- `--start-date value`: Creation date. Optional. (format: `Jan 1 15:04:05 2011`).
|
|
||||||
- `--duration value`: Duration which the certificate is valid for. Optional. (default: 8760h0m0s)
|
|
||||||
- `--ca`: If provided, this cert generates it's own certificate authority. Optional.
|
|
||||||
- Examples:
|
|
||||||
- `gitea cert --host git.example.com,example.com,www.example.com --ca`
|
|
||||||
|
|
||||||
### dump
|
|
||||||
|
|
||||||
Dumps all files and databases into a zip file. Outputs into a file like `gitea-dump-1482906742.zip`
|
|
||||||
in the current directory.
|
|
||||||
|
|
||||||
- Options:
|
|
||||||
- `--file name`, `-f name`: Name of the dump file with will be created. Optional. (default: gitea-dump-[timestamp].zip).
|
|
||||||
- `--tempdir path`, `-t path`: Path to the temporary directory used. Optional. (default: /tmp).
|
|
||||||
- `--skip-repository`, `-R`: Skip the repository dumping. Optional.
|
|
||||||
- `--skip-custom-dir`: Skip dumping of the custom dir. Optional.
|
|
||||||
- `--skip-lfs-data`: Skip dumping of LFS data. Optional.
|
|
||||||
- `--skip-attachment-data`: Skip dumping of attachment data. Optional.
|
|
||||||
- `--skip-package-data`: Skip dumping of package data. Optional.
|
|
||||||
- `--skip-log`: Skip dumping of log data. Optional.
|
|
||||||
- `--database`, `-d`: Specify the database SQL syntax. Optional (supported arguments: sqlite3, mysql, mssql, postgres).
|
|
||||||
- `--verbose`, `-V`: If provided, shows additional details. Optional.
|
|
||||||
- `--type`: Set the dump output format. Optional. (formats: zip, tar, tar.sz, tar.gz, tar.xz, tar.bz2, tar.br, tar.lz4, tar.zst default: zip).
|
|
||||||
- Examples:
|
|
||||||
- `gitea dump`
|
|
||||||
- `gitea dump --verbose`
|
|
||||||
|
|
||||||
### generate
|
|
||||||
|
|
||||||
Generates random values and tokens for usage in configuration file. Useful for generating values
|
|
||||||
for automatic deployments.
|
|
||||||
|
|
||||||
- Commands:
|
|
||||||
- `secret`:
|
|
||||||
- Options:
|
|
||||||
- `INTERNAL_TOKEN`: Token used for an internal API call authentication.
|
|
||||||
- `JWT_SECRET`: LFS & OAUTH2 JWT authentication secret (LFS_JWT_SECRET is aliased to this option for backwards compatibility).
|
|
||||||
- `SECRET_KEY`: Global secret key.
|
|
||||||
- Examples:
|
|
||||||
- `gitea generate secret INTERNAL_TOKEN`
|
|
||||||
- `gitea generate secret JWT_SECRET`
|
|
||||||
- `gitea generate secret SECRET_KEY`
|
|
||||||
|
|
||||||
### keys
|
|
||||||
|
|
||||||
Provides an SSHD AuthorizedKeysCommand. Needs to be configured in the sshd config file:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
...
|
|
||||||
# The value of -e and the AuthorizedKeysCommandUser should match the
|
|
||||||
# username running Gitea
|
|
||||||
AuthorizedKeysCommandUser git
|
|
||||||
AuthorizedKeysCommand /path/to/gitea keys -e git -u %u -t %t -k %k
|
|
||||||
```
|
|
||||||
|
|
||||||
The command will return the appropriate authorized_keys line for the
|
|
||||||
provided key. You should also set the value
|
|
||||||
`SSH_CREATE_AUTHORIZED_KEYS_FILE=false` in the `[server]` section of
|
|
||||||
`app.ini`.
|
|
||||||
|
|
||||||
NB: opensshd requires the Gitea program to be owned by root and not
|
|
||||||
writable by group or others. The program must be specified by an absolute
|
|
||||||
path.
|
|
||||||
NB: Gitea must be running for this command to succeed.
|
|
||||||
|
|
||||||
### migrate
|
|
||||||
|
|
||||||
Migrates the database. This command can be used to run other commands before starting the server for the first time.
|
|
||||||
This command is idempotent.
|
|
||||||
|
|
||||||
### doctor check
|
|
||||||
|
|
||||||
Diagnose and potentially fix problems with the current Gitea instance.
|
|
||||||
Several checks are run by default, but additional ones can be run:
|
|
||||||
|
|
||||||
- `gitea doctor check --list` - will list all the available checks
|
|
||||||
- `gitea doctor check --all` - will run all available checks
|
|
||||||
- `gitea doctor check --default` - will run the default checks
|
|
||||||
- `gitea doctor check --run [check(s),]...` - will run the named checks
|
|
||||||
|
|
||||||
Some problems can be automatically fixed by passing the `--fix` option.
|
|
||||||
Extra logging can be set with `--log-file=...`.
|
|
||||||
|
|
||||||
#### doctor recreate-table
|
|
||||||
|
|
||||||
Sometimes when there are migrations the old columns and default values may be left
|
|
||||||
unchanged in the database schema. This may lead to warning such as:
|
|
||||||
|
|
||||||
```
|
|
||||||
2020/08/02 11:32:29 ...rm/session_schema.go:360:Sync() [W] Table user Column keep_activity_private db default is , struct default is 0
|
|
||||||
```
|
|
||||||
|
|
||||||
You can cause Gitea to recreate these tables and copy the old data into the new table
|
|
||||||
with the defaults set appropriately by using:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea doctor recreate-table user
|
|
||||||
```
|
|
||||||
|
|
||||||
You can ask Gitea to recreate multiple tables using:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea doctor recreate-table table1 table2 ...
|
|
||||||
```
|
|
||||||
|
|
||||||
And if you would like Gitea to recreate all tables simply call:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea doctor recreate-table
|
|
||||||
```
|
|
||||||
|
|
||||||
It is highly recommended to back-up your database before running these commands.
|
|
||||||
|
|
||||||
### doctor convert
|
|
||||||
|
|
||||||
Converts a MySQL database from utf8 to utf8mb4 or a MSSQL database from varchar to nvarchar.
|
|
||||||
|
|
||||||
### manager
|
|
||||||
|
|
||||||
Manage running server operations:
|
|
||||||
|
|
||||||
- Commands:
|
|
||||||
- `shutdown`: Gracefully shutdown the running process
|
|
||||||
- `restart`: Gracefully restart the running process - (not implemented for windows servers)
|
|
||||||
- `flush-queues`: Flush queues in the running process
|
|
||||||
- Options:
|
|
||||||
- `--timeout value`: Timeout for the flushing process (default: 1m0s)
|
|
||||||
- `--non-blocking`: Set to true to not wait for flush to complete before returning
|
|
||||||
- `logging`: Adjust logging commands
|
|
||||||
- Commands:
|
|
||||||
- `pause`: Pause logging
|
|
||||||
- Notes:
|
|
||||||
- The logging level will be raised to INFO temporarily if it is below this level.
|
|
||||||
- Gitea will buffer logs up to a certain point and will drop them after that point.
|
|
||||||
- `resume`: Resume logging
|
|
||||||
- `release-and-reopen`: Cause Gitea to release and re-open files and connections used for logging (Equivalent to sending SIGUSR1 to Gitea.)
|
|
||||||
- `remove name`: Remove the named logger
|
|
||||||
- Options:
|
|
||||||
- `--group group`, `-g group`: Set the group to remove the sublogger from. (defaults to `default`)
|
|
||||||
- `add`: Add a logger
|
|
||||||
- Commands:
|
|
||||||
- `console`: Add a console logger
|
|
||||||
- Options:
|
|
||||||
- `--group value`, `-g value`: Group to add logger to - will default to "default"
|
|
||||||
- `--name value`, `-n value`: Name of the new logger - will default to mode
|
|
||||||
- `--level value`, `-l value`: Logging level for the new logger
|
|
||||||
- `--stacktrace-level value`, `-L value`: Stacktrace logging level
|
|
||||||
- `--flags value`, `-F value`: Flags for the logger
|
|
||||||
- `--expression value`, `-e value`: Matching expression for the logger
|
|
||||||
- `--prefix value`, `-p value`: Prefix for the logger
|
|
||||||
- `--color`: Use color in the logs
|
|
||||||
- `--stderr`: Output console logs to stderr - only relevant for console
|
|
||||||
- `file`: Add a file logger
|
|
||||||
- Options:
|
|
||||||
- `--group value`, `-g value`: Group to add logger to - will default to "default"
|
|
||||||
- `--name value`, `-n value`: Name of the new logger - will default to mode
|
|
||||||
- `--level value`, `-l value`: Logging level for the new logger
|
|
||||||
- `--stacktrace-level value`, `-L value`: Stacktrace logging level
|
|
||||||
- `--flags value`, `-F value`: Flags for the logger
|
|
||||||
- `--expression value`, `-e value`: Matching expression for the logger
|
|
||||||
- `--prefix value`, `-p value`: Prefix for the logger
|
|
||||||
- `--color`: Use color in the logs
|
|
||||||
- `--filename value`, `-f value`: Filename for the logger -
|
|
||||||
- `--rotate`, `-r`: Rotate logs
|
|
||||||
- `--max-size value`, `-s value`: Maximum size in bytes before rotation
|
|
||||||
- `--daily`, `-d`: Rotate logs daily
|
|
||||||
- `--max-days value`, `-D value`: Maximum number of daily logs to keep
|
|
||||||
- `--compress`, `-z`: Compress rotated logs
|
|
||||||
- `--compression-level value`, `-Z value`: Compression level to use
|
|
||||||
- `conn`: Add a network connection logger
|
|
||||||
- Options:
|
|
||||||
- `--group value`, `-g value`: Group to add logger to - will default to "default"
|
|
||||||
- `--name value`, `-n value`: Name of the new logger - will default to mode
|
|
||||||
- `--level value`, `-l value`: Logging level for the new logger
|
|
||||||
- `--stacktrace-level value`, `-L value`: Stacktrace logging level
|
|
||||||
- `--flags value`, `-F value`: Flags for the logger
|
|
||||||
- `--expression value`, `-e value`: Matching expression for the logger
|
|
||||||
- `--prefix value`, `-p value`: Prefix for the logger
|
|
||||||
- `--color`: Use color in the logs
|
|
||||||
- `--reconnect-on-message`, `-R`: Reconnect to host for every message
|
|
||||||
- `--reconnect`, `-r`: Reconnect to host when connection is dropped
|
|
||||||
- `--protocol value`, `-P value`: Set protocol to use: tcp, unix, or udp (defaults to tcp)
|
|
||||||
- `--address value`, `-a value`: Host address and port to connect to (defaults to :7020)
|
|
||||||
- `smtp`: Add an SMTP logger
|
|
||||||
- Options:
|
|
||||||
- `--group value`, `-g value`: Group to add logger to - will default to "default"
|
|
||||||
- `--name value`, `-n value`: Name of the new logger - will default to mode
|
|
||||||
- `--level value`, `-l value`: Logging level for the new logger
|
|
||||||
- `--stacktrace-level value`, `-L value`: Stacktrace logging level
|
|
||||||
- `--flags value`, `-F value`: Flags for the logger
|
|
||||||
- `--expression value`, `-e value`: Matching expression for the logger
|
|
||||||
- `--prefix value`, `-p value`: Prefix for the logger
|
|
||||||
- `--color`: Use color in the logs
|
|
||||||
- `--username value`, `-u value`: Mail server username
|
|
||||||
- `--password value`, `-P value`: Mail server password
|
|
||||||
- `--host value`, `-H value`: Mail server host (defaults to: 127.0.0.1:25)
|
|
||||||
- `--send-to value`, `-s value`: Email address(es) to send to
|
|
||||||
- `--subject value`, `-S value`: Subject header of sent emails
|
|
||||||
- `processes`: Display Gitea processes and goroutine information
|
|
||||||
- Options:
|
|
||||||
- `--flat`: Show processes as flat table rather than as tree
|
|
||||||
- `--no-system`: Do not show system processes
|
|
||||||
- `--stacktraces`: Show stacktraces for goroutines associated with processes
|
|
||||||
- `--json`: Output as json
|
|
||||||
- `--cancel PID`: Send cancel to process with PID. (Only for non-system processes.)
|
|
||||||
|
|
||||||
### dump-repo
|
|
||||||
|
|
||||||
Dump-repo dumps repository data from Git/GitHub/Gitea/GitLab:
|
|
||||||
|
|
||||||
- Options:
|
|
||||||
- `--git_service service` : Git service, it could be `git`, `github`, `gitea`, `gitlab`, If clone_addr could be recognized, this could be ignored.
|
|
||||||
- `--repo_dir dir`, `-r dir`: Repository dir path to store the data
|
|
||||||
- `--clone_addr addr`: The URL will be clone, currently could be a git/github/gitea/gitlab http/https URL. i.e. https://github.com/lunny/tango.git
|
|
||||||
- `--auth_username lunny`: The username to visit the clone_addr
|
|
||||||
- `--auth_password <password>`: The password to visit the clone_addr
|
|
||||||
- `--auth_token <token>`: The personal token to visit the clone_addr
|
|
||||||
- `--owner_name lunny`: The data will be stored on a directory with owner name if not empty
|
|
||||||
- `--repo_name tango`: The data will be stored on a directory with repository name if not empty
|
|
||||||
- `--units <units>`: Which items will be migrated, one or more units should be separated as comma. wiki, issues, labels, releases, release_assets, milestones, pull_requests, comments are allowed. Empty means all units.
|
|
||||||
|
|
||||||
### restore-repo
|
|
||||||
|
|
||||||
Restore-repo restore repository data from disk dir:
|
|
||||||
|
|
||||||
- Options:
|
|
||||||
- `--repo_dir dir`, `-r dir`: Repository dir path to restore from
|
|
||||||
- `--owner_name lunny`: Restore destination owner name
|
|
||||||
- `--repo_name tango`: Restore destination repository name
|
|
||||||
- `--units <units>`: Which items will be restored, one or more units should be separated as comma. wiki, issues, labels, releases, release_assets, milestones, pull_requests, comments are allowed. Empty means all units.
|
|
||||||
|
|
||||||
### actions generate-runner-token
|
|
||||||
|
|
||||||
Generate a new token for a runner to use to register with the server
|
|
||||||
|
|
||||||
- Options:
|
|
||||||
- `--scope {owner}[/{repo}]`, `-s {owner}[/{repo}]`: To limit the scope of the runner, no scope means the runner can be used for all repos, but you can also limit it to a specific repo or owner
|
|
||||||
|
|
||||||
To register a global runner:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea actions generate-runner-token
|
|
||||||
```
|
|
||||||
|
|
||||||
To register a runner for a specific organization, in this case `org`:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea actions generate-runner-token -s org
|
|
||||||
```
|
|
||||||
|
|
||||||
To register a runner for a specific repo, in this case `username/test-repo`:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea actions generate-runner-token -s username/test-repo
|
|
||||||
```
|
|
@ -1,542 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2023-05-23T09:00:00+08:00"
|
|
||||||
title: "Gitea 命令行"
|
|
||||||
slug: "command-line"
|
|
||||||
sidebar_position: 1
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/command-line
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Gitea 命令行"
|
|
||||||
sidebar_position: 1
|
|
||||||
identifier: "command-line"
|
|
||||||
---
|
|
||||||
|
|
||||||
# 命令行
|
|
||||||
|
|
||||||
## 用法
|
|
||||||
|
|
||||||
`gitea [全局选项] 命令 [命令或全局选项] [参数...]`
|
|
||||||
|
|
||||||
## 全局选项
|
|
||||||
|
|
||||||
所有全局选项均可被放置在命令级别。
|
|
||||||
|
|
||||||
- `--help`,`-h`:显示帮助文本并退出。可选。
|
|
||||||
- `--version`,`-v`:显示版本信息并退出。可选。 (示例:`Gitea version 1.1.0+218-g7b907ed built with: bindata, sqlite`)。
|
|
||||||
- `--custom-path path`,`-C path`:Gitea 自定义文件夹的路径。可选。 (默认值:`AppWorkPath`/custom 或 `$GITEA_CUSTOM`)。
|
|
||||||
- `--config path`,`-c path`:Gitea 配置文件的路径。可选。 (默认值:`custom`/conf/app.ini)。
|
|
||||||
- `--work-path path`,`-w path`:Gitea 的 `AppWorkPath`。可选。 (默认值:LOCATION_OF_GITEA_BINARY 或 `$GITEA_WORK_DIR`)
|
|
||||||
|
|
||||||
注意:默认的 custom-path、config 和 work-path 也可以在构建时更改(如果需要)。
|
|
||||||
|
|
||||||
## 命令
|
|
||||||
|
|
||||||
### web
|
|
||||||
|
|
||||||
启动服务器:
|
|
||||||
|
|
||||||
- 选项:
|
|
||||||
- `--port number`,`-p number`:端口号。可选。 (默认值:3000)。覆盖配置文件中的设置。
|
|
||||||
- `--install-port number`:运行安装页面的端口号。可选。 (默认值:3000)。覆盖配置文件中的设置。
|
|
||||||
- `--pid path`,`-P path`:Pid 文件的路径。可选。
|
|
||||||
- `--quiet`,`-q`:只在控制台上输出 Fatal 日志,用于在设置日志之前发出的日志。
|
|
||||||
- `--verbose`:在控制台上输出跟踪日志,用于在设置日志之前发出的日志。
|
|
||||||
- 示例:
|
|
||||||
- `gitea web`
|
|
||||||
- `gitea web --port 80`
|
|
||||||
- `gitea web --config /etc/gitea.ini --pid /some/custom/gitea.pid`
|
|
||||||
- 注意:
|
|
||||||
- Gitea 不应以 root 用户身份运行。要绑定到低于 1024 的端口,您可以在 Linux 上使用 setcap 命令:`sudo setcap 'cap_net_bind_service=+ep' /path/to/gitea`。每次更新 Gitea 都需要重新执行此操作。
|
|
||||||
|
|
||||||
### admin
|
|
||||||
|
|
||||||
管理员操作:
|
|
||||||
|
|
||||||
- 命令:
|
|
||||||
- `user`:
|
|
||||||
- `list`:
|
|
||||||
- 选项:
|
|
||||||
- `--admin`:仅列出管理员用户。可选。
|
|
||||||
- 描述:列出所有现有用户。
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin user list`
|
|
||||||
- `delete`:
|
|
||||||
- 选项:
|
|
||||||
- `--email`:要删除的用户的电子邮件。
|
|
||||||
- `--username`:要删除的用户的用户名。
|
|
||||||
- `--id`:要删除的用户的ID。
|
|
||||||
- 必须提供 `--id`、`--username` 或 `--email` 中的一个。如果提供多个,则所有条件必须匹配。
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin user delete --id 1`
|
|
||||||
- `create`:
|
|
||||||
- 选项:
|
|
||||||
- `--name value`:用户名。必填。自 Gitea 1.9.0 版本起,请改用 `--username` 标志。
|
|
||||||
- `--username value`:用户名。必填。Gitea 1.9.0 新增。
|
|
||||||
- `--password value`:密码。必填。
|
|
||||||
- `--email value`:邮箱。必填。
|
|
||||||
- `--admin`:如果提供此选项,将创建一个管理员用户。可选。
|
|
||||||
- `--access-token`:如果提供,将为用户创建访问令牌。可选。(默认值:false)。
|
|
||||||
- `--must-change-password`:如果提供,创建的用户将在初始登录后需要选择一个新密码。可选。(默认值:true)。
|
|
||||||
- `--random-password`:如果提供,将使用随机生成的密码作为创建用户的密码。`--password` 的值将被忽略。可选。
|
|
||||||
- `--random-password-length`:如果提供,将用于配置随机生成密码的长度。可选。(默认值:12)
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin user create --username myname --password asecurepassword --email me@example.com`
|
|
||||||
- `change-password`:
|
|
||||||
- 选项:
|
|
||||||
- `--username value`,`-u value`:用户名。必填。
|
|
||||||
- `--password value`,`-p value`:新密码。必填。
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin user change-password --username myname --password asecurepassword`
|
|
||||||
- `must-change-password`:
|
|
||||||
- 参数:
|
|
||||||
- `[username...]`:需要更改密码的用户
|
|
||||||
- 选项:
|
|
||||||
- `--all`,`-A`:强制所有用户更改密码
|
|
||||||
- `--exclude username`,`-e username`:排除给定的用户。可以多次设置。
|
|
||||||
- `--unset`:撤销对给定用户的强制密码更改
|
|
||||||
- `regenerate`:
|
|
||||||
- 选项:
|
|
||||||
- `hooks`:重新生成所有仓库的 Git Hooks。
|
|
||||||
- `keys`:重新生成 authorized_keys 文件。
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin regenerate hooks`
|
|
||||||
- `gitea admin regenerate keys`
|
|
||||||
- `auth`:
|
|
||||||
- `list`:
|
|
||||||
- 描述:列出所有存在的外部认证源。
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin auth list`
|
|
||||||
- `delete`:
|
|
||||||
- 选项:
|
|
||||||
- `--id`:要删除的源的 ID。必填。
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin auth delete --id 1`
|
|
||||||
- `add-oauth`:
|
|
||||||
- 选项:
|
|
||||||
- `--name`:应用程序名称。
|
|
||||||
- `--provider`:OAuth2 提供者。
|
|
||||||
- `--key`:客户端 ID(Key)。
|
|
||||||
- `--secret`:客户端密钥。
|
|
||||||
- `--auto-discover-url`:OpenID Connect 自动发现 URL(仅在使用 OpenID Connect 作为提供程序时需要)。
|
|
||||||
- `--use-custom-urls`:在 GitLab/GitHub OAuth 端点上使用自定义 URL。
|
|
||||||
- `--custom-tenant-id`:在 OAuth 端点上使用自定义租户 ID。
|
|
||||||
- `--custom-auth-url`:使用自定义授权 URL(GitLab/GitHub 的选项)。
|
|
||||||
- `--custom-token-url`:使用自定义令牌 URL(GitLab/GitHub 的选项)。
|
|
||||||
- `--custom-profile-url`:使用自定义配置文件 URL(GitLab/GitHub 的选项)。
|
|
||||||
- `--custom-email-url`:使用自定义电子邮件 URL(GitHub 的选项)。
|
|
||||||
- `--icon-url`:OAuth2 登录源的自定义图标 URL。
|
|
||||||
- `--skip-local-2fa`:允许源覆盖本地 2FA。(可选)
|
|
||||||
- `--scopes`:请求此 OAuth2 源的附加范围。(可选)
|
|
||||||
- `--required-claim-name`:必须设置的声明名称,以允许用户使用此源登录。(可选)
|
|
||||||
- `--required-claim-value`:必须设置的声明值,以允许用户使用此源登录。(可选)
|
|
||||||
- `--group-claim-name`:提供此源的组名的声明名称。(可选)
|
|
||||||
- `--admin-group`:管理员用户的组声明值。(可选)
|
|
||||||
- `--restricted-group`:受限用户的组声明值。(可选)
|
|
||||||
- `--group-team-map`:组与组织团队之间的 JSON 映射。(可选)
|
|
||||||
- `--group-team-map-removal`:根据组自动激活团队成员资格的删除。(可选)
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin auth add-oauth --name external-github --provider github --key OBTAIN_FROM_SOURCE --secret OBTAIN_FROM_SOURCE`
|
|
||||||
- `update-oauth`:
|
|
||||||
- 选项:
|
|
||||||
- `--id`:要更新的源的 ID。必填。
|
|
||||||
- `--name`:应用程序名称。
|
|
||||||
- `--provider`:OAuth2 提供者。
|
|
||||||
- `--key`:客户端 ID(Key)。
|
|
||||||
- `--secret`:客户端密钥。
|
|
||||||
- `--auto-discover-url`:OpenID Connect 自动发现 URL(仅在使用 OpenID Connect 作为提供程序时需要)。
|
|
||||||
- `--use-custom-urls`:在 GitLab/GitHub OAuth 端点上使用自定义 URL。
|
|
||||||
- `--custom-tenant-id`:在 OAuth 端点上使用自定义租户 ID。
|
|
||||||
- `--custom-auth-url`:使用自定义授权 URL(GitLab/GitHub 的选项)。
|
|
||||||
- `--custom-token-url`:使用自定义令牌 URL(GitLab/GitHub 的选项)。
|
|
||||||
- `--custom-profile-url`:使用自定义配置文件 URL(GitLab/GitHub 的选项)。
|
|
||||||
- `--custom-email-url`:使用自定义电子邮件 URL(GitHub 的选项)。
|
|
||||||
- `--icon-url`:OAuth2 登录源的自定义图标 URL。
|
|
||||||
- `--skip-local-2fa`:允许源覆盖本地 2FA。(可选)
|
|
||||||
- `--scopes`:请求此 OAuth2 源的附加范围。
|
|
||||||
- `--required-claim-name`:必须设置的声明名称,以允许用户使用此源登录。(可选)
|
|
||||||
- `--required-claim-value`:必须设置的声明值,以允许用户使用此源登录。(可选)
|
|
||||||
- `--group-claim-name`:提供此源的组名的声明名称。(可选)
|
|
||||||
- `--admin-group`:管理员用户的组声明值。(可选)
|
|
||||||
- `--restricted-group`:受限用户的组声明值。(可选)
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin auth update-oauth --id 1 --name external-github-updated`
|
|
||||||
- `add-smtp`:
|
|
||||||
- 选项:
|
|
||||||
- `--name`:应用程序名称。必填。
|
|
||||||
- `--auth-type`:SMTP 认证类型(PLAIN/LOGIN/CRAM-MD5)。默认为 PLAIN。
|
|
||||||
- `--host`:SMTP 主机。必填。
|
|
||||||
- `--port`:SMTP 端口。必填。
|
|
||||||
- `--force-smtps`:SMTPS 始终在端口 465 上使用。设置此选项以强制在其他端口上使用 SMTPS。
|
|
||||||
- `--skip-verify`:跳过 TLS 验证。
|
|
||||||
- `--helo-hostname`:发送 HELO 时使用的主机名。留空以发送当前主机名。
|
|
||||||
- `--disable-helo`:禁用 SMTP helo。
|
|
||||||
- `--allowed-domains`:留空以允许所有域。使用逗号(',')分隔多个域。
|
|
||||||
- `--skip-local-2fa`:跳过 2FA 登录。
|
|
||||||
- `--active`:启用此认证源。
|
|
||||||
备注:
|
|
||||||
`--force-smtps`、`--skip-verify`、`--disable-helo`、`--skip-local-2fs` 和 `--active` 选项可以采用以下形式使用:
|
|
||||||
- `--option`、`--option=true` 以启用选项
|
|
||||||
- `--option=false` 以禁用选项
|
|
||||||
如果未指定这些选项,则在 `update-smtp` 中不会更改值,或者在 `add-smtp` 中将使用默认的 `false` 值。
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin auth add-smtp --name ldap --host smtp.mydomain.org --port 587 --skip-verify --active`
|
|
||||||
- `update-smtp`:
|
|
||||||
- 选项:
|
|
||||||
- `--id`:要更新的源的 ID。必填。
|
|
||||||
- 其他选项与 `add-smtp` 共享
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin auth update-smtp --id 1 --host smtp.mydomain.org --port 587 --skip-verify=false`
|
|
||||||
- `gitea admin auth update-smtp --id 1 --active=false`
|
|
||||||
- `add-ldap`:添加新的 LDAP(通过 Bind DN)认证源
|
|
||||||
- 选项:
|
|
||||||
- `--name value`:认证名称。必填。
|
|
||||||
- `--not-active`:停用认证源。
|
|
||||||
- `--security-protocol value`:安全协议名称。必填。
|
|
||||||
- `--skip-tls-verify`:禁用 TLS 验证。
|
|
||||||
- `--host value`:LDAP 服务器的地址。必填。
|
|
||||||
- `--port value`:连接到 LDAP 服务器时使用的端口。必填。
|
|
||||||
- `--user-search-base value`:用户帐户将在其中搜索的 LDAP 基础路径。必填。
|
|
||||||
- `--user-filter value`:声明如何查找试图进行身份验证的用户记录的 LDAP 过滤器。必填。
|
|
||||||
- `--admin-filter value`:指定是否应授予用户管理员特权的 LDAP 过滤器。
|
|
||||||
- `--restricted-filter value`:指定是否应将用户设置为受限状态的 LDAP 过滤器。
|
|
||||||
- `--username-attribute value`:用户 LDAP 记录中包含用户名的属性。
|
|
||||||
- `--firstname-attribute value`:用户 LDAP 记录中包含用户名字的属性。
|
|
||||||
- `--surname-attribute value`:用户 LDAP 记录中包含用户姓氏的属性。
|
|
||||||
- `--email-attribute value`:用户 LDAP 记录中包含用户电子邮件地址的属性。必填。
|
|
||||||
- `--public-ssh-key-attribute value`:用户 LDAP 记录中包含用户公共 SSH 密钥的属性。
|
|
||||||
- `--avatar-attribute value`:用户 LDAP 记录中包含用户头像的属性。
|
|
||||||
- `--bind-dn value`:在搜索用户时绑定到 LDAP 服务器的 DN。
|
|
||||||
- `--bind-password value`:绑定 DN 的密码(如果有)。
|
|
||||||
- `--attributes-in-bind`:在绑定 DN 上下文中获取属性。
|
|
||||||
- `--synchronize-users`:启用用户同步。
|
|
||||||
- `--page-size value`:搜索页面大小。
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin auth add-ldap --name ldap --security-protocol unencrypted --host mydomain.org --port 389 --user-search-base "ou=Users,dc=mydomain,dc=org" --user-filter "(&(objectClass=posixAccount)(|(uid=%[1]s)(mail=%[1]s)))" --email-attribute mail`
|
|
||||||
- `update-ldap`:更新现有的 LDAP(通过 Bind DN)认证源
|
|
||||||
- 选项:
|
|
||||||
- `--id value`:认证源的 ID。必填。
|
|
||||||
- `--name value`:认证名称。
|
|
||||||
- `--not-active`:停用认证源。
|
|
||||||
- `--security-protocol value`:安全协议名称。
|
|
||||||
- `--skip-tls-verify`:禁用 TLS 验证。
|
|
||||||
- `--host value`:LDAP 服务器的地址。
|
|
||||||
- `--port value`:连接到 LDAP 服务器时使用的端口。
|
|
||||||
- `--user-search-base value`:用户帐户将在其中搜索的 LDAP 基础路径。
|
|
||||||
- `--user-filter value`:声明如何查找试图进行身份验证的用户记录的 LDAP 过滤器。
|
|
||||||
- `--admin-filter value`:指定是否应授予用户管理员特权的 LDAP 过滤器。
|
|
||||||
- `--restricted-filter value`:指定是否应将用户设置为受限状态的 LDAP 过滤器。
|
|
||||||
- `--username-attribute value`:用户 LDAP 记录中包含用户名的属性。
|
|
||||||
- `--firstname-attribute value`:用户 LDAP 记录中包含用户名字的属性。
|
|
||||||
- `--surname-attribute value`:用户 LDAP 记录中包含用户姓氏的属性。
|
|
||||||
- `--email-attribute value`:用户 LDAP 记录中包含用户电子邮件地址的属性。
|
|
||||||
- `--public-ssh-key-attribute value`:用户 LDAP 记录中包含用户公共 SSH 密钥的属性。
|
|
||||||
- `--avatar-attribute value`:用户 LDAP 记录中包含用户头像的属性。
|
|
||||||
- `--bind-dn value`:在搜索用户时绑定到 LDAP 服务器的 DN。
|
|
||||||
- `--bind-password value`:绑定 DN 的密码(如果有)。
|
|
||||||
- `--attributes-in-bind`:在绑定 DN 上下文中获取属性。
|
|
||||||
- `--synchronize-users`:启用用户同步。
|
|
||||||
- `--page-size value`:搜索页面大小。
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin auth update-ldap --id 1 --name "my ldap auth source"`
|
|
||||||
- `gitea admin auth update-ldap --id 1 --username-attribute uid --firstname-attribute givenName --surname-attribute sn`
|
|
||||||
- `add-ldap-simple`:添加新的 LDAP(简单身份验证)认证源
|
|
||||||
- 选项:
|
|
||||||
- `--name value`:认证名称。必填。
|
|
||||||
- `--not-active`:停用认证源。
|
|
||||||
- `--security-protocol value`:安全协议名称。必填。
|
|
||||||
- `--skip-tls-verify`:禁用 TLS 验证。
|
|
||||||
- `--host value`:LDAP 服务器的地址。必填。
|
|
||||||
- `--port value`:连接到 LDAP 服务器时使用的端口。必填。
|
|
||||||
- `--user-search-base value`:用户帐户将在其中搜索的 LDAP 基础路径。
|
|
||||||
- `--user-filter value`:声明如何查找试图进行身份验证的用户记录的 LDAP 过滤器。必填。
|
|
||||||
- `--admin-filter value`:指定是否应授予用户管理员特权的 LDAP 过滤器。
|
|
||||||
- `--restricted-filter value`:指定是否应将用户设置为受限状态的 LDAP 过滤器。
|
|
||||||
- `--username-attribute value`:用户 LDAP 记录中包含用户名的属性。
|
|
||||||
- `--firstname-attribute value`:用户 LDAP 记录中包含用户名字的属性。
|
|
||||||
- `--surname-attribute value`:用户 LDAP 记录中包含用户姓氏的属性。
|
|
||||||
- `--email-attribute value`:用户 LDAP 记录中包含用户电子邮件地址的属性。必填。
|
|
||||||
- `--public-ssh-key-attribute value`:用户 LDAP 记录中包含用户公共 SSH 密钥的属性。
|
|
||||||
- `--avatar-attribute value`:用户 LDAP 记录中包含用户头像的属性。
|
|
||||||
- `--user-dn value`:用户的 DN。必填。
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin auth add-ldap-simple --name ldap --security-protocol unencrypted --host mydomain.org --port 389 --user-dn "cn=%s,ou=Users,dc=mydomain,dc=org" --user-filter "(&(objectClass=posixAccount)(cn=%s))" --email-attribute mail`
|
|
||||||
- `update-ldap-simple`:更新现有的 LDAP(简单身份验证)认证源
|
|
||||||
- 选项:
|
|
||||||
- `--id value`:认证源的 ID。必填。
|
|
||||||
- `--name value`:认证名称。
|
|
||||||
- `--not-active`:停用认证源。
|
|
||||||
- `--security-protocol value`:安全协议名称。
|
|
||||||
- `--skip-tls-verify`:禁用 TLS 验证。
|
|
||||||
- `--host value`:LDAP 服务器的地址。
|
|
||||||
- `--port value`:连接到 LDAP 服务器时使用的端口。
|
|
||||||
- `--user-search-base value`:用户帐户将在其中搜索的 LDAP 基础路径。
|
|
||||||
- `--user-filter value`:声明如何查找试图进行身份验证的用户记录的 LDAP 过滤器。
|
|
||||||
- `--admin-filter value`:指定是否应授予用户管理员特权的 LDAP 过滤器。
|
|
||||||
- `--restricted-filter value`:指定是否应将用户设置为受限状态的 LDAP 过滤器。
|
|
||||||
- `--username-attribute value`:用户 LDAP 记录中包含用户名的属性。
|
|
||||||
- `--firstname-attribute value`:用户 LDAP 记录中包含用户名字的属性。
|
|
||||||
- `--surname-attribute value`:用户 LDAP 记录中包含用户姓氏的属性。
|
|
||||||
- `--email-attribute value`:用户 LDAP 记录中包含用户电子邮件地址的属性。
|
|
||||||
- `--public-ssh-key-attribute value`:用户 LDAP 记录中包含用户公共 SSH 密钥的属性。
|
|
||||||
- `--avatar-attribute value`:用户 LDAP 记录中包含用户头像的属性。
|
|
||||||
- `--user-dn value`:用户的 DN。
|
|
||||||
- 示例:
|
|
||||||
- `gitea admin auth update-ldap-simple --id 1 --name "my ldap auth source"`
|
|
||||||
- `gitea admin auth update-ldap-simple --id 1 --username-attribute uid --firstname-attribute givenName --surname-attribute sn`
|
|
||||||
|
|
||||||
### cert
|
|
||||||
|
|
||||||
生成自签名的SSL证书。将输出到当前目录下的`cert.pem`和`key.pem`文件中,并且会覆盖任何现有文件。
|
|
||||||
|
|
||||||
- 选项:
|
|
||||||
- `--host value`:逗号分隔的主机名和IP地址列表,此证书适用于这些主机。支持使用通配符。必填。
|
|
||||||
- `--ecdsa-curve value`:用于生成密钥的ECDSA曲线。可选。有效选项为P224、P256、P384、P521。
|
|
||||||
- `--rsa-bits value`:要生成的RSA密钥的大小。可选。如果设置了--ecdsa-curve,则忽略此选项。(默认值:3072)。
|
|
||||||
- `--start-date value`:证书的创建日期。可选。(格式:`Jan 1 15:04:05 2011`)。
|
|
||||||
- `--duration value`:证书有效期。可选。(默认值:8760h0m0s)
|
|
||||||
- `--ca`:如果提供此选项,则证书将生成自己的证书颁发机构。可选。
|
|
||||||
- 示例:
|
|
||||||
- `gitea cert --host git.example.com,example.com,www.example.com --ca`
|
|
||||||
|
|
||||||
### dump
|
|
||||||
|
|
||||||
将所有文件和数据库导出到一个zip文件中。输出文件将保存在当前目录下,类似于`gitea-dump-1482906742.zip`。
|
|
||||||
|
|
||||||
- 选项:
|
|
||||||
- `--file name`,`-f name`:指定要创建的导出文件的名称。可选。(默认值:gitea-dump-[timestamp].zip)。
|
|
||||||
- `--tempdir path`,`-t path`:指定临时目录的路径。可选。(默认值:/tmp)。
|
|
||||||
- `--skip-repository`,`-R`:跳过仓库的导出。可选。
|
|
||||||
- `--skip-custom-dir`:跳过自定义目录的导出。可选。
|
|
||||||
- `--skip-lfs-data`:跳过LFS数据的导出。可选。
|
|
||||||
- `--skip-attachment-data`:跳过附件数据的导出。可选。
|
|
||||||
- `--skip-package-data`:跳过包数据的导出。可选。
|
|
||||||
- `--skip-log`:跳过日志数据的导出。可选。
|
|
||||||
- `--database`,`-d`:指定数据库的SQL语法。可选。
|
|
||||||
- `--verbose`,`-V`:如果提供此选项,显示附加详细信息。可选。
|
|
||||||
- `--type`:设置导出的格式。可选。(默认值:zip)
|
|
||||||
- 示例:
|
|
||||||
- `gitea dump`
|
|
||||||
- `gitea dump --verbose`
|
|
||||||
|
|
||||||
### generate
|
|
||||||
|
|
||||||
用于在配置文件中生成随机值和令牌。对于自动部署时生成值非常有用。
|
|
||||||
|
|
||||||
- 命令:
|
|
||||||
- `secret`:
|
|
||||||
- 选项:
|
|
||||||
- `INTERNAL_TOKEN`: 用于内部 API 调用身份验证的令牌。
|
|
||||||
- `JWT_SECRET`: 用于 LFS 和 OAUTH2 JWT 身份验证的密钥(LFS_JWT_SECRET 是此选项的别名,用于向后兼容)。
|
|
||||||
- `SECRET_KEY`: 全局密钥。
|
|
||||||
- 示例:
|
|
||||||
- `gitea generate secret INTERNAL_TOKEN`
|
|
||||||
- `gitea generate secret JWT_SECRET`
|
|
||||||
- `gitea generate secret SECRET_KEY`
|
|
||||||
|
|
||||||
### keys
|
|
||||||
|
|
||||||
提供一个 SSHD AuthorizedKeysCommand。需要在 sshd 配置文件中进行配置:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
...
|
|
||||||
# -e 的值和 AuthorizedKeysCommandUser 应与运行 Gitea 的用户名匹配
|
|
||||||
AuthorizedKeysCommandUser git
|
|
||||||
AuthorizedKeysCommand /path/to/gitea keys -e git -u %u -t %t -k %k
|
|
||||||
```
|
|
||||||
|
|
||||||
该命令将返回适用于提供的密钥的合适 authorized_keys 行。您还应在 `app.ini` 的 `[server]` 部分设置值 `SSH_CREATE_AUTHORIZED_KEYS_FILE=false`。
|
|
||||||
|
|
||||||
注意: opensshd 要求 Gitea 程序由 root 拥有,并且不可由组或其他人写入。程序必须使用绝对路径指定。
|
|
||||||
注意: Gitea 必须在运行此命令时处于运行状态才能成功。
|
|
||||||
|
|
||||||
### migrate
|
|
||||||
|
|
||||||
迁移数据库。该命令可用于在首次启动服务器之前运行其他命令。此命令是幂等的。
|
|
||||||
|
|
||||||
### doctor check
|
|
||||||
|
|
||||||
对 Gitea 实例进行诊断,可以修复一些可修复的问题。
|
|
||||||
默认只运行部分检查,额外的检查可以参考:
|
|
||||||
|
|
||||||
- `gitea doctor check --list` - 列出所有可用的检查
|
|
||||||
- `gitea doctor check --all` - 运行所有可用的检查
|
|
||||||
- `gitea doctor check --default` - 运行默认的检查
|
|
||||||
- `gitea doctor check --run [check(s),]...` - 运行指定的名字的检查
|
|
||||||
|
|
||||||
有些问题可以通过设置 `--fix` 选项进行自动修复。
|
|
||||||
额外的日志可以通过 `--log-file=...` 进行设置。
|
|
||||||
|
|
||||||
#### doctor recreate-table
|
|
||||||
|
|
||||||
有时,在迁移时,旧的列和默认值可能会在数据库模式中保持不变。这可能会导致警告,如下所示:
|
|
||||||
|
|
||||||
```
|
|
||||||
2020/08/02 11:32:29 ...rm/session_schema.go:360:Sync() [W] Table user Column keep_activity_private db default is , struct default is 0
|
|
||||||
```
|
|
||||||
|
|
||||||
您可以通过以下方式让 Gitea 重新创建这些表,并将旧数据复制到新表中,并适当设置默认值:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea doctor recreate-table user
|
|
||||||
```
|
|
||||||
|
|
||||||
您可以使用以下方式让 Gitea 重新创建多个表:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea doctor recreate-table table1 table2 ...
|
|
||||||
```
|
|
||||||
|
|
||||||
如果您希望 Gitea 重新创建所有表,请直接调用:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea doctor recreate-table
|
|
||||||
```
|
|
||||||
|
|
||||||
强烈建议在运行这些命令之前备份您的数据库。
|
|
||||||
|
|
||||||
### doctor convert
|
|
||||||
|
|
||||||
将现有的 MySQL 数据库从 utf8 转换为 utf8mb4,或者把 MSSQL 数据库从 varchar 转换为 nvarchar。
|
|
||||||
|
|
||||||
### manager
|
|
||||||
|
|
||||||
管理运行中的服务器操作:
|
|
||||||
|
|
||||||
- 命令:
|
|
||||||
- `shutdown`: 优雅地关闭运行中的进程
|
|
||||||
- `restart`: 优雅地重新启动运行中的进程(对于Windows服务器尚未实现)
|
|
||||||
- `flush-queues`: 刷新运行中的进程中的队列
|
|
||||||
- 选项:
|
|
||||||
- `--timeout value`: 刷新过程的超时时间(默认值: 1m0s)
|
|
||||||
- `--non-blocking`: 设置为true,以在返回之前不等待刷新完成
|
|
||||||
- `logging`: 调整日志命令
|
|
||||||
- 命令:
|
|
||||||
- `pause`: 暂停日志记录
|
|
||||||
- 注意:
|
|
||||||
- 如果日志级别低于此级别,日志级别将被临时提升为INFO。
|
|
||||||
- Gitea将在一定程度上缓冲日志,并在超过该点后丢弃日志。
|
|
||||||
- `resume`: 恢复日志记录
|
|
||||||
- `release-and-reopen`: 使Gitea释放和重新打开用于日志记录的文件和连接(相当于向Gitea发送SIGUSR1信号)。
|
|
||||||
- `remove name`: 删除指定的日志记录器
|
|
||||||
- 选项:
|
|
||||||
- `--group group`, `-g group`: 从中删除子记录器的组(默认为`default`)
|
|
||||||
- `add`: 添加日志记录器
|
|
||||||
- 命令:
|
|
||||||
- `console`: 添加控制台日志记录器
|
|
||||||
- 选项:
|
|
||||||
- `--group value`, `-g value`: 要添加日志记录器的组 - 默认为"default"
|
|
||||||
- `--name value`, `-n value`: 新日志记录器的名称 - 默认为模式
|
|
||||||
- `--level value`, `-l value`: 新日志记录器的日志级别
|
|
||||||
- `--stacktrace-level value`, `-L value`: 堆栈跟踪日志级别
|
|
||||||
- `--flags value`, `-F value`: 日志记录器的标志
|
|
||||||
- `--expression value`, `-e value`: 日志记录器的匹配表达式
|
|
||||||
- `--prefix value`, `-p value`: 日志记录器的前缀
|
|
||||||
- `--color`: 在日志中使用颜色
|
|
||||||
- `--stderr`: 将控制台日志输出到stderr - 仅适用于控制台
|
|
||||||
- `file`: 添加文件日志记录器
|
|
||||||
- 选项:
|
|
||||||
- `--group value`, `-g value`: 要添加日志记录器的组 - 默认为"default"
|
|
||||||
- `--name value`, `-n value`: 新日志记录器的名称 - 默认为模式
|
|
||||||
- `--level value`, `-l value`: 新日志记录器的日志级别
|
|
||||||
- `--stacktrace-level value`, `-L value`: 堆栈跟踪日志级别
|
|
||||||
- `--flags value`, `-F value`: 日志记录器的标志
|
|
||||||
- `--expression value`, `-e value`: 日志记录器的匹配表达式
|
|
||||||
- `--prefix value`, `-p value`: 日志记录器的前缀
|
|
||||||
- `--color`: 在日志中使用颜色
|
|
||||||
- `--filename value`, `-f value`: 日志记录器的文件名
|
|
||||||
- `--rotate`, `-r`: 轮转日志
|
|
||||||
- `--max-size value`, `-s value`: 在轮转之前的最大大小(以字节为单位)
|
|
||||||
- `--daily`, `-d`: 每天轮转日志
|
|
||||||
- `--max-days value`, `-D value`: 保留的每日日志的最大数量
|
|
||||||
- `--compress`, `-z`: 压缩轮转的日志
|
|
||||||
- `--compression-level value`, `-Z value`: 使用的压缩级别
|
|
||||||
- `conn`: 添加网络连接日志记录器
|
|
||||||
- 选项:
|
|
||||||
- `--group value`, `-g value`: 要添加日志记录器的组 - 默认为"default"
|
|
||||||
- `--name value`, `-n value`: 新日志记录器的名称 - 默认为模式
|
|
||||||
- `--level value`, `-l value`: 新日志记录器的日志级别
|
|
||||||
- `--stacktrace-level value`, `-L value`: 堆栈跟踪日志级别
|
|
||||||
- `--flags value`, `-F value`: 日志记录器的标志
|
|
||||||
- `--expression value`, `-e value`: 日志记录器的匹配表达式
|
|
||||||
- `--prefix value`, `-p value`: 日志记录器的前缀
|
|
||||||
- `--color`: 在日志中使用颜色
|
|
||||||
- `--reconnect-on-message`, `-R`: 对于每个消息重新连接主机
|
|
||||||
- `--reconnect`, `-r`: 连接中断时重新连接主机
|
|
||||||
- `--protocol value`, `-P value`: 设置要使用的协议:tcp、unix或udp(默认为tcp)
|
|
||||||
- `--address value`, `-a value`: 要连接到的主机地址和端口(默认为:7020)
|
|
||||||
- `smtp`: 添加SMTP日志记录器
|
|
||||||
- 选项:
|
|
||||||
- `--group value`, `-g value`: 要添加日志记录器的组 - 默认为"default"
|
|
||||||
- `--name value`, `-n value`: 新日志记录器的名称 - 默认为模式
|
|
||||||
- `--level value`, `-l value`: 新日志记录器的日志级别
|
|
||||||
- `--stacktrace-level value`, `-L value`: 堆栈跟踪日志级别
|
|
||||||
- `--flags value`, `-F value`: 日志记录器的标志
|
|
||||||
- `--expression value`, `-e value`: 日志记录器的匹配表达式
|
|
||||||
- `--prefix value`, `-p value`: 日志记录器的前缀
|
|
||||||
- `--color`: 在日志中使用颜色
|
|
||||||
- `--username value`, `-u value`: 邮件服务器用户名
|
|
||||||
- `--password value`, `-P value`: 邮件服务器密码
|
|
||||||
- `--host value`, `-H value`: 邮件服务器主机(默认为: 127.0.0.1:25)
|
|
||||||
- `--send-to value`, `-s value`: 要发送到的电子邮件地址
|
|
||||||
- `--subject value`, `-S value`: 发送电子邮件的主题标题
|
|
||||||
- `processes`: 显示 Gitea 进程和 Goroutine 信息
|
|
||||||
- 选项:
|
|
||||||
- `--flat`: 以平面表格形式显示进程,而不是树形结构
|
|
||||||
- `--no-system`: 不显示系统进程
|
|
||||||
- `--stacktraces`: 显示与进程关联的 Goroutine 的堆栈跟踪
|
|
||||||
- `--json`: 输出为 JSON 格式
|
|
||||||
- `--cancel PID`: 向具有 PID 的进程发送取消命令(仅适用于非系统进程)
|
|
||||||
|
|
||||||
### dump-repo
|
|
||||||
|
|
||||||
`dump-repo` 从 Git/GitHub/Gitea/GitLab 中转储存储库数据:
|
|
||||||
|
|
||||||
- 选项:
|
|
||||||
- `--git_service service`:Git 服务,可以是 `git`、`github`、`gitea`、`gitlab`。如果 `clone_addr` 可以被识别,则可以忽略此选项。
|
|
||||||
- `--repo_dir dir`,`-r dir`:存储数据的存储库目录路径。
|
|
||||||
- `--clone_addr addr`:将被克隆的 URL,目前可以是 git/github/gitea/gitlab 的 http/https URL。例如:https://github.com/lunny/tango.git
|
|
||||||
- `--auth_username lunny`:访问 `clone_addr` 的用户名。
|
|
||||||
- `--auth_password <password>`:访问 `clone_addr` 的密码。
|
|
||||||
- `--auth_token <token>`:访问 `clone_addr` 的个人令牌。
|
|
||||||
- `--owner_name lunny`:如果非空,数据将存储在具有所有者名称的目录中。
|
|
||||||
- `--repo_name tango`:如果非空,数据将存储在具有存储库名称的目录中。
|
|
||||||
- `--units <units>`:要迁移的项目,一个或多个项目应以逗号分隔。允许的项目有 wiki, issues, labels, releases, release_assets, milestones, pull_requests, comments。如果为空,则表示所有项目。
|
|
||||||
|
|
||||||
### restore-repo
|
|
||||||
|
|
||||||
`restore-repo` 从磁盘目录中还原存储库数据:
|
|
||||||
|
|
||||||
- 选项:
|
|
||||||
- `--repo_dir dir`,`-r dir`:还原数据的存储库目录路径。
|
|
||||||
- `--owner_name lunny`:还原目标所有者名称。
|
|
||||||
- `--repo_name tango`:还原目标存储库名称。
|
|
||||||
- `--units <units>`:要还原的项目,一个或多个项目应以逗号分隔。允许的项目有 wiki, issues, labels, releases, release_assets, milestones, pull_requests, comments。如果为空,则表示所有项目。
|
|
||||||
|
|
||||||
### actions generate-runner-token
|
|
||||||
|
|
||||||
生成一个供 Runner 使用的新令牌,用于向服务器注册。
|
|
||||||
|
|
||||||
- 选项:
|
|
||||||
- `--scope {owner}[/{repo}]`,`-s {owner}[/{repo}]`:限制 Runner 的范围,没有范围表示该 Runner 可用于所有仓库,但你也可以将其限制为特定的仓库或所有者。
|
|
||||||
|
|
||||||
要注册全局 Runner:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea actions generate-runner-token
|
|
||||||
```
|
|
||||||
|
|
||||||
要注册特定组织的 Runner,例如 `org`:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea actions generate-runner-token -s org
|
|
||||||
```
|
|
||||||
|
|
||||||
要注册特定仓库的 Runner,例如 `username/test-repo`:
|
|
||||||
|
|
||||||
```
|
|
||||||
gitea actions generate-runner-token -s username/test-repo
|
|
||||||
```
|
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -1,403 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2017-04-15T14:56:00+02:00"
|
|
||||||
title: "Customizing Gitea"
|
|
||||||
slug: "customizing-gitea"
|
|
||||||
sidebar_position: 100
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/customizing-gitea
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Customizing Gitea"
|
|
||||||
identifier: "customizing-gitea"
|
|
||||||
sidebar_position: 100
|
|
||||||
---
|
|
||||||
|
|
||||||
# Customizing Gitea
|
|
||||||
|
|
||||||
Customizing Gitea is typically done using the `CustomPath` folder - by default this is
|
|
||||||
the `custom` folder from the working directory (WorkPath), but may be different if your build has
|
|
||||||
set this differently. This is the central place to override configuration settings,
|
|
||||||
templates, etc. You can check the `CustomPath` using `gitea help`. You can also find
|
|
||||||
the path on the _Configuration_ tab in the _Site Administration_ page. You can override
|
|
||||||
the `CustomPath` by setting either the `GITEA_CUSTOM` environment variable or by
|
|
||||||
using the `--custom-path` option on the `gitea` binary. (The option will override the
|
|
||||||
environment variable.)
|
|
||||||
|
|
||||||
If Gitea is deployed from binary, all default paths will be relative to the Gitea
|
|
||||||
binary. If installed from a distribution, these paths will likely be modified to
|
|
||||||
the Linux Filesystem Standard. Gitea will attempt to create required folders, including
|
|
||||||
`custom/`. Distributions may provide a symlink for `custom` using `/etc/gitea/`.
|
|
||||||
|
|
||||||
Application settings can be found in file `CustomConf` which is by default,
|
|
||||||
`$GITEA_CUSTOM/conf/app.ini` but may be different if your build has set this differently.
|
|
||||||
Again `gitea help` will allow you review this variable and you can override it using the
|
|
||||||
`--config` option on the `gitea` binary.
|
|
||||||
|
|
||||||
- [Quick Cheat Sheet](administration/config-cheat-sheet.md)
|
|
||||||
- [Complete List](https://github.com/go-gitea/gitea/blob/main/custom/conf/app.example.ini)
|
|
||||||
|
|
||||||
If the `CustomPath` folder can't be found despite checking `gitea help`, check the `GITEA_CUSTOM`
|
|
||||||
environment variable; this can be used to override the default path to something else.
|
|
||||||
`GITEA_CUSTOM` might, for example, be set by an init script. You can check whether the value
|
|
||||||
is set under the "Configuration" tab on the site administration page.
|
|
||||||
|
|
||||||
- [List of Environment Variables](administration/environment-variables.md)
|
|
||||||
|
|
||||||
**Note:** Gitea must perform a full restart to see configuration changes.
|
|
||||||
|
|
||||||
## Serving custom public files
|
|
||||||
|
|
||||||
To make Gitea serve custom public files (like pages and images), use the folder
|
|
||||||
`$GITEA_CUSTOM/public/` as the webroot. Symbolic links will be followed.
|
|
||||||
At the moment, only the following files are served:
|
|
||||||
|
|
||||||
- `public/robots.txt`
|
|
||||||
- files in the `public/.well-known/` folder
|
|
||||||
- files in the `public/assets/` folder
|
|
||||||
|
|
||||||
For example, a file `image.png` stored in `$GITEA_CUSTOM/public/assets/`, can be accessed with
|
|
||||||
the url `http://gitea.domain.tld/assets/image.png`.
|
|
||||||
|
|
||||||
## Changing the logo
|
|
||||||
|
|
||||||
To build a custom logo and/or favicon clone the Gitea source repository, replace `assets/logo.svg` and/or `assets/favicon.svg` and run
|
|
||||||
`make generate-images`. `assets/favicon.svg` is used for the favicon only. This will update below output files which you can then place in `$GITEA_CUSTOM/public/assets/img` on your server:
|
|
||||||
|
|
||||||
- `public/assets/img/logo.svg` - Used for site icon, app icon
|
|
||||||
- `public/assets/img/logo.png` - Used for Open Graph
|
|
||||||
- `public/assets/img/avatar_default.png` - Used as the default avatar image
|
|
||||||
- `public/assets/img/apple-touch-icon.png` - Used on iOS devices for bookmarks
|
|
||||||
- `public/assets/img/favicon.svg` - Used for favicon
|
|
||||||
- `public/assets/img/favicon.png` - Used as fallback for browsers that don't support SVG favicons
|
|
||||||
|
|
||||||
In case the source image is not in vector format, you can attempt to convert a raster image using tools like [this](https://www.aconvert.com/image/png-to-svg/).
|
|
||||||
|
|
||||||
## Customizing Gitea pages and resources
|
|
||||||
|
|
||||||
Gitea's executable contains all the resources required to run: templates, images, style-sheets
|
|
||||||
and translations. Any of them can be overridden by placing a replacement in a matching path
|
|
||||||
inside the `custom` directory. For example, to replace the default `.gitignore` provided
|
|
||||||
for C++ repositories, we want to replace `options/gitignore/C++`. To do this, a replacement
|
|
||||||
must be placed in `$GITEA_CUSTOM/options/gitignore/C++` (see about the location of the `CustomPath`
|
|
||||||
directory at the top of this document).
|
|
||||||
|
|
||||||
Every single page of Gitea can be changed. Dynamic content is generated using [go templates](https://pkg.go.dev/html/template),
|
|
||||||
which can be modified by placing replacements below the `$GITEA_CUSTOM/templates` directory.
|
|
||||||
|
|
||||||
To obtain any embedded file (including templates), the [`gitea embedded` tool](administration/cmd-embedded.md) can be used. Alternatively, they can be found in the [`templates`](https://github.com/go-gitea/gitea/tree/main/templates) directory of Gitea source (Note: the example link is from the `main` branch. Make sure to use templates compatible with the release you are using).
|
|
||||||
|
|
||||||
Be aware that any statement contained inside `{{` and `}}` are Gitea's template syntax and
|
|
||||||
shouldn't be touched without fully understanding these components.
|
|
||||||
|
|
||||||
### Customizing startpage / homepage
|
|
||||||
|
|
||||||
Copy [`home.tmpl`](https://github.com/go-gitea/gitea/blob/main/templates/home.tmpl) for your version of Gitea from `templates` to `$GITEA_CUSTOM/templates`.
|
|
||||||
Edit as you wish.
|
|
||||||
Dont forget to restart your Gitea to apply the changes.
|
|
||||||
|
|
||||||
### Adding links and tabs
|
|
||||||
|
|
||||||
If all you want is to add extra links to the top navigation bar or footer, or extra tabs to the repository view, you can put them in `extra_links.tmpl` (links added to the navbar), `extra_links_footer.tmpl` (links added to the left side of footer), and `extra_tabs.tmpl` inside your `$GITEA_CUSTOM/templates/custom/` directory.
|
|
||||||
|
|
||||||
For instance, let's say you are in Germany and must add the famously legally-required "Impressum"/about page, listing who is responsible for the site's content:
|
|
||||||
just place it under your "$GITEA_CUSTOM/public/assets/" directory (for instance `$GITEA_CUSTOM/public/assets/impressum.html`) and put a link to it in either `$GITEA_CUSTOM/templates/custom/extra_links.tmpl` or `$GITEA_CUSTOM/templates/custom/extra_links_footer.tmpl`.
|
|
||||||
|
|
||||||
To match the current style, the link should have the class name "item", and you can use `{{AppSubUrl}}` to get the base URL:
|
|
||||||
`<a class="item" href="{{AppSubUrl}}/assets/impressum.html">Impressum</a>`
|
|
||||||
|
|
||||||
For more information, see [Adding Legal Pages](administration/adding-legal-pages.md).
|
|
||||||
|
|
||||||
You can add new tabs in the same way, putting them in `extra_tabs.tmpl`.
|
|
||||||
The exact HTML needed to match the style of other tabs is in the file
|
|
||||||
`templates/repo/header.tmpl`
|
|
||||||
([source in GitHub](https://github.com/go-gitea/gitea/blob/main/templates/repo/header.tmpl))
|
|
||||||
|
|
||||||
### Other additions to the page
|
|
||||||
|
|
||||||
Apart from `extra_links.tmpl` and `extra_tabs.tmpl`, there are other useful templates you can put in your `$GITEA_CUSTOM/templates/custom/` directory:
|
|
||||||
|
|
||||||
- `header.tmpl`, just before the end of the `<head>` tag where you can add custom CSS files for instance.
|
|
||||||
- `body_outer_pre.tmpl`, right after the start of `<body>`.
|
|
||||||
- `body_inner_pre.tmpl`, before the top navigation bar, but already inside the main container `<div class="full height">`.
|
|
||||||
- `body_inner_post.tmpl`, before the end of the main container.
|
|
||||||
- `body_outer_post.tmpl`, before the bottom `<footer>` element.
|
|
||||||
- `footer.tmpl`, right before the end of the `<body>` tag, a good place for additional JavaScript.
|
|
||||||
|
|
||||||
### Using Gitea variables
|
|
||||||
|
|
||||||
It's possible to use various Gitea variables in your custom templates.
|
|
||||||
|
|
||||||
First, _temporarily_ enable development mode: in your `app.ini` change from `RUN_MODE = prod` to `RUN_MODE = dev`. Then add `{{ $ | DumpVar }}` to any of your templates, restart Gitea and refresh that page; that will dump all available variables.
|
|
||||||
|
|
||||||
Find the data that you need, and use the corresponding variable; for example, if you need the name of the repository then you'd use `{{.Repository.Name}}`.
|
|
||||||
|
|
||||||
If you need to transform that data somehow, and aren't familiar with Go, an easy workaround is to add the data to the DOM and add a small JavaScript script block to manipulate the data.
|
|
||||||
|
|
||||||
### Example: PlantUML
|
|
||||||
|
|
||||||
You can add [PlantUML](https://plantuml.com/) support to Gitea's markdown by using a PlantUML server.
|
|
||||||
The data is encoded and sent to the PlantUML server which generates the picture. There is an online
|
|
||||||
demo server at http://www.plantuml.com/plantuml, but if you (or your users) have sensitive data you
|
|
||||||
can set up your own [PlantUML server](https://plantuml.com/server) instead. To set up PlantUML rendering,
|
|
||||||
copy JavaScript files from https://gitea.com/davidsvantesson/plantuml-code-highlight and put them in your
|
|
||||||
`$GITEA_CUSTOM/public/assets/` folder. Then add the following to `custom/footer.tmpl`:
|
|
||||||
|
|
||||||
```html
|
|
||||||
<script>
|
|
||||||
$(async () => {
|
|
||||||
if (!$('.language-plantuml').length) return;
|
|
||||||
await Promise.all([
|
|
||||||
$.getScript('https://your-gitea-server.com/assets/deflate.js'),
|
|
||||||
$.getScript('https://your-gitea-server.com/assets/encode.js'),
|
|
||||||
$.getScript('https://your-gitea-server.com/assets/plantuml_codeblock_parse.js'),
|
|
||||||
]);
|
|
||||||
// Replace call with address to your plantuml server
|
|
||||||
parsePlantumlCodeBlocks("https://www.plantuml.com/plantuml");
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
```
|
|
||||||
|
|
||||||
You can then add blocks like the following to your markdown:
|
|
||||||
|
|
||||||
```plantuml
|
|
||||||
Alice -> Bob: Authentication Request
|
|
||||||
Bob --> Alice: Authentication Response
|
|
||||||
|
|
||||||
Alice -> Bob: Another authentication Request
|
|
||||||
Alice <-- Bob: Another authentication Response
|
|
||||||
```
|
|
||||||
|
|
||||||
The script will detect tags with `class="language-plantuml"`, but you can change this by providing a second argument to `parsePlantumlCodeBlocks`.
|
|
||||||
|
|
||||||
### Example: STL Preview
|
|
||||||
|
|
||||||
You can display STL file directly in Gitea by adding:
|
|
||||||
|
|
||||||
```html
|
|
||||||
<script>
|
|
||||||
function lS(src) {
|
|
||||||
return new Promise(function (resolve, reject) {
|
|
||||||
let s = document.createElement("script");
|
|
||||||
s.src = src;
|
|
||||||
s.addEventListener("load", () => {
|
|
||||||
resolve();
|
|
||||||
});
|
|
||||||
document.body.appendChild(s);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($('.view-raw>a[href$=".stl" i]').length) {
|
|
||||||
$("body").append(
|
|
||||||
'<link href="/assets/Madeleine.js/src/css/Madeleine.css" rel="stylesheet">'
|
|
||||||
);
|
|
||||||
Promise.all([
|
|
||||||
lS("/assets/Madeleine.js/src/lib/stats.js"),
|
|
||||||
lS("/assets/Madeleine.js/src/lib/detector.js"),
|
|
||||||
lS("/assets/Madeleine.js/src/lib/three.min.js"),
|
|
||||||
lS("/assets/Madeleine.js/src/Madeleine.js"),
|
|
||||||
]).then(function () {
|
|
||||||
$(".view-raw")
|
|
||||||
.attr("id", "view-raw")
|
|
||||||
.attr("style", "padding: 0;margin-bottom: -10px;");
|
|
||||||
new Madeleine({
|
|
||||||
target: "view-raw",
|
|
||||||
data: $('.view-raw>a[href$=".stl" i]').attr("href"),
|
|
||||||
path: "/assets/Madeleine.js/src",
|
|
||||||
});
|
|
||||||
$('.view-raw>a[href$=".stl"]').remove();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
```
|
|
||||||
|
|
||||||
to the file `templates/custom/footer.tmpl`
|
|
||||||
|
|
||||||
You also need to download the content of the library [Madeleine.js](https://github.com/beige90/Madeleine.js) and place it under `$GITEA_CUSTOM/public/assets/` folder.
|
|
||||||
|
|
||||||
You should end-up with a folder structure similar to:
|
|
||||||
|
|
||||||
```
|
|
||||||
$GITEA_CUSTOM/templates
|
|
||||||
-- custom
|
|
||||||
`-- footer.tmpl
|
|
||||||
|
|
||||||
$GITEA_CUSTOM/public/assets/
|
|
||||||
-- Madeleine.js
|
|
||||||
|-- LICENSE
|
|
||||||
|-- README.md
|
|
||||||
|-- css
|
|
||||||
| |-- pygment_trac.css
|
|
||||||
| `-- stylesheet.css
|
|
||||||
|-- examples
|
|
||||||
| |-- ajax.html
|
|
||||||
| |-- index.html
|
|
||||||
| `-- upload.html
|
|
||||||
|-- images
|
|
||||||
| |-- bg_hr.png
|
|
||||||
| |-- blacktocat.png
|
|
||||||
| |-- icon_download.png
|
|
||||||
| `-- sprite_download.png
|
|
||||||
|-- models
|
|
||||||
| |-- dino2.stl
|
|
||||||
| |-- ducati.stl
|
|
||||||
| |-- gallardo.stl
|
|
||||||
| |-- lamp.stl
|
|
||||||
| |-- octocat.stl
|
|
||||||
| |-- skull.stl
|
|
||||||
| `-- treefrog.stl
|
|
||||||
`-- src
|
|
||||||
|-- Madeleine.js
|
|
||||||
|-- css
|
|
||||||
| `-- Madeleine.css
|
|
||||||
|-- icons
|
|
||||||
| |-- logo.png
|
|
||||||
| |-- madeleine.eot
|
|
||||||
| |-- madeleine.svg
|
|
||||||
| |-- madeleine.ttf
|
|
||||||
| `-- madeleine.woff
|
|
||||||
`-- lib
|
|
||||||
|-- MadeleineConverter.js
|
|
||||||
|-- MadeleineLoader.js
|
|
||||||
|-- detector.js
|
|
||||||
|-- stats.js
|
|
||||||
`-- three.min.js
|
|
||||||
```
|
|
||||||
|
|
||||||
Then restart Gitea and open a STL file on your Gitea instance.
|
|
||||||
|
|
||||||
## Customizing Gitea mails
|
|
||||||
|
|
||||||
The `$GITEA_CUSTOM/templates/mail` folder allows changing the body of every mail of Gitea.
|
|
||||||
Templates to override can be found in the
|
|
||||||
[`templates/mail`](https://github.com/go-gitea/gitea/tree/main/templates/mail)
|
|
||||||
directory of Gitea source.
|
|
||||||
Override by making a copy of the file under `$GITEA_CUSTOM/templates/mail` using a
|
|
||||||
full path structure matching source.
|
|
||||||
|
|
||||||
Any statement contained inside `{{` and `}}` are Gitea's template
|
|
||||||
syntax and shouldn't be touched without fully understanding these components.
|
|
||||||
|
|
||||||
## Adding Analytics to Gitea
|
|
||||||
|
|
||||||
Google Analytics, Matomo (previously Piwik), and other analytics services can be added to Gitea. To add the tracking code, refer to the `Other additions to the page` section of this document, and add the JavaScript to the `$GITEA_CUSTOM/templates/custom/header.tmpl` file.
|
|
||||||
|
|
||||||
## Customizing gitignores, labels, licenses, locales, and readmes
|
|
||||||
|
|
||||||
Place custom files in corresponding sub-folder under `custom/options`.
|
|
||||||
|
|
||||||
**NOTE:** The files should not have a file extension, e.g. `Labels` rather than `Labels.txt`
|
|
||||||
|
|
||||||
### gitignores
|
|
||||||
|
|
||||||
To add custom .gitignore, add a file with existing [.gitignore rules](https://git-scm.com/docs/gitignore) in it to `$GITEA_CUSTOM/options/gitignore`
|
|
||||||
|
|
||||||
## Customizing the git configuration
|
|
||||||
|
|
||||||
Starting with Gitea 1.20, you can customize the git configuration via the `git.config` section.
|
|
||||||
|
|
||||||
### Enabling signed git pushes
|
|
||||||
|
|
||||||
To enable signed git pushes, set these two options:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[git.config]
|
|
||||||
receive.advertisePushOptions = true
|
|
||||||
receive.certNonceSeed = <randomstring>
|
|
||||||
```
|
|
||||||
|
|
||||||
`certNonceSeed` should be set to a random string and be kept secret.
|
|
||||||
|
|
||||||
### Labels
|
|
||||||
|
|
||||||
Starting with Gitea 1.19, you can add a file that follows the [YAML label format](https://github.com/go-gitea/gitea/blob/main/options/label/Advanced.yaml) to `$GITEA_CUSTOM/options/label`:
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
labels:
|
|
||||||
- name: "foo/bar" # name of the label that will appear in the dropdown
|
|
||||||
exclusive: true # whether to use the exclusive namespace for scoped labels. scoped delimiter is /
|
|
||||||
color: aabbcc # hex colour coding
|
|
||||||
description: Some label # long description of label intent
|
|
||||||
```
|
|
||||||
|
|
||||||
The [legacy file format](https://github.com/go-gitea/gitea/blob/main/options/label/Default) can still be used following the format below, however we strongly recommend using the newer YAML format instead.
|
|
||||||
|
|
||||||
`#hex-color label name ; label description`
|
|
||||||
|
|
||||||
For more information, see the [labels documentation](usage/labels.md).
|
|
||||||
|
|
||||||
### Licenses
|
|
||||||
|
|
||||||
To add a custom license, add a file with the license text to `$GITEA_CUSTOM/options/license`
|
|
||||||
|
|
||||||
### Locales
|
|
||||||
|
|
||||||
Locales are managed via our [Crowdin](https://crowdin.com/project/gitea).
|
|
||||||
You can override a locale by placing an altered locale file in `$GITEA_CUSTOM/options/locale`.
|
|
||||||
Gitea's default locale files can be found in the [`options/locale`](https://github.com/go-gitea/gitea/tree/main/options/locale) source folder and these should be used as examples for your changes.
|
|
||||||
|
|
||||||
To add a completely new locale, as well as placing the file in the above location, you will need to add the new lang and name to the `[i18n]` section in your `app.ini`. Keep in mind that Gitea will use those settings as **overrides**, so if you want to keep the other languages as well you will need to copy/paste the default values and add your own to them.
|
|
||||||
|
|
||||||
```
|
|
||||||
[i18n]
|
|
||||||
LANGS = en-US,foo-BAR
|
|
||||||
NAMES = English,FooBar
|
|
||||||
```
|
|
||||||
|
|
||||||
The first locale will be used as the default if user browser's language doesn't match any locale in the list.
|
|
||||||
|
|
||||||
Locales may change between versions, so keeping track of your customized locales is highly encouraged.
|
|
||||||
|
|
||||||
### Readmes
|
|
||||||
|
|
||||||
To add a custom Readme, add a markdown formatted file (without an `.md` extension) to `$GITEA_CUSTOM/options/readme`
|
|
||||||
|
|
||||||
**NOTE:** readme templates support **variable expansion**.
|
|
||||||
currently there are `{Name}` (name of repository), `{Description}`, `{CloneURL.SSH}`, `{CloneURL.HTTPS}` and `{OwnerName}`
|
|
||||||
|
|
||||||
### Reactions
|
|
||||||
|
|
||||||
To change reaction emoji's you can set allowed reactions at app.ini
|
|
||||||
|
|
||||||
```
|
|
||||||
[ui]
|
|
||||||
REACTIONS = +1, -1, laugh, confused, heart, hooray, eyes
|
|
||||||
```
|
|
||||||
|
|
||||||
A full list of supported emoji's is at [emoji list](https://gitea.com/gitea/gitea.com/issues/8)
|
|
||||||
|
|
||||||
## Customizing the look of Gitea
|
|
||||||
|
|
||||||
The built-in themes are `gitea-light`, `gitea-dark`, and `gitea-auto` (which automatically adapts to OS settings).
|
|
||||||
|
|
||||||
The default theme can be changed via `DEFAULT_THEME` in the [ui](administration/config-cheat-sheet.md#ui-ui) section of `app.ini`.
|
|
||||||
|
|
||||||
Gitea also has support for user themes, which means every user can select which theme should be used.
|
|
||||||
The list of themes a user can choose from can be configured with the `THEMES` value in the [ui](administration/config-cheat-sheet.md#ui-ui) section of `app.ini`.
|
|
||||||
|
|
||||||
To make a custom theme available to all users:
|
|
||||||
|
|
||||||
1. Add a CSS file to `$GITEA_CUSTOM/public/assets/css/theme-<theme-name>.css`.
|
|
||||||
The value of `$GITEA_CUSTOM` of your instance can be queried by calling `gitea help` and looking up the value of "CustomPath".
|
|
||||||
2. Add `<theme-name>` to the comma-separated list of setting `THEMES` in `app.ini`
|
|
||||||
|
|
||||||
Community themes are listed in [gitea/awesome-gitea#themes](https://gitea.com/gitea/awesome-gitea#themes).
|
|
||||||
|
|
||||||
The default theme sources can be found [here](https://github.com/go-gitea/gitea/blob/main/web_src/css/themes).
|
|
||||||
|
|
||||||
If your custom theme is considered a dark theme, set the global css variable `--is-dark-theme` to `true`.
|
|
||||||
This allows Gitea to adjust the Monaco code editor's theme accordingly.
|
|
||||||
|
|
||||||
## Customizing fonts
|
|
||||||
|
|
||||||
Fonts can be customized using CSS variables:
|
|
||||||
|
|
||||||
```css
|
|
||||||
:root {
|
|
||||||
--fonts-proportional: /* custom proportional fonts */ !important;
|
|
||||||
--fonts-monospace: /* custom monospace fonts */ !important;
|
|
||||||
--fonts-emoji: /* custom emoji fonts */ !important;
|
|
||||||
}
|
|
||||||
```
|
|
@ -1,91 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2017-04-15T14:56:00+02:00"
|
|
||||||
title: "自定义 Gitea 配置"
|
|
||||||
slug: "customizing-gitea"
|
|
||||||
sidebar_position: 100
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/customizing-gitea
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "自定义 Gitea 配置"
|
|
||||||
sidebar_position: 100
|
|
||||||
identifier: "customizing-gitea"
|
|
||||||
---
|
|
||||||
|
|
||||||
# 自定义 Gitea 配置
|
|
||||||
|
|
||||||
Gitea 引用 `custom` 目录中的自定义配置文件来覆盖配置、模板等默认配置。
|
|
||||||
|
|
||||||
如果从二进制部署 Gitea ,则所有默认路径都将相对于该 gitea 二进制文件;如果从发行版安装,则可能会将这些路径修改为Linux文件系统标准。Gitea
|
|
||||||
将会自动创建包括 `custom/` 在内的必要应用目录,应用本身的配置存放在
|
|
||||||
`custom/conf/app.ini` 当中。在发行版中可能会以 `/etc/gitea/` 的形式为 `custom` 设置一个符号链接,查看配置详情请移步:
|
|
||||||
|
|
||||||
- [快速备忘单](administration/config-cheat-sheet.md)
|
|
||||||
- [完整配置清单](https://github.com/go-gitea/gitea/blob/main/custom/conf/app.example.ini)
|
|
||||||
|
|
||||||
如果您在 binary 同目录下无法找到 `custom` 文件夹,请检查您的 `GITEA_CUSTOM`
|
|
||||||
环境变量配置, 因为它可能被配置到了其他地方(可能被一些启动脚本设置指定了目录)。
|
|
||||||
|
|
||||||
- [环境变量清单](administration/environment-variables.md)
|
|
||||||
|
|
||||||
**注:** 必须完全重启 Gitea 以使配置生效。
|
|
||||||
|
|
||||||
## 使用自定义 /robots.txt
|
|
||||||
|
|
||||||
将 [想要展示的内容](http://www.robotstxt.org/) 存放在 `custom` 目录中的
|
|
||||||
`robots.txt` 文件来让 Gitea 使用自定义的`/robots.txt` (默认:空 404)。
|
|
||||||
|
|
||||||
## 使用自定义的公共文件
|
|
||||||
|
|
||||||
将自定义的公共文件(比如页面和图片)作为 webroot 放在 `custom/public/` 中来让 Gitea 提供这些自定义内容(符号链接将被追踪)。
|
|
||||||
|
|
||||||
举例说明:`image.png` 存放在 `custom/public/assets/`中,那么它可以通过链接 http://gitea.domain.tld/assets/image.png 访问。
|
|
||||||
|
|
||||||
## 修改默认头像
|
|
||||||
|
|
||||||
替换以下目录中的 png 图片: `custom/public/assets/img/avatar\_default.png`
|
|
||||||
|
|
||||||
## 自定义 Gitea 页面
|
|
||||||
|
|
||||||
您可以改变 Gitea `custom/templates` 的每个单页面。您可以在 Gitea 源码的 `templates` 目录中找到用于覆盖的模板文件,应用将根据
|
|
||||||
`custom/templates` 目录下的路径结构进行匹配和覆盖。
|
|
||||||
|
|
||||||
包含在 `{{` 和 `}}` 中的任何语句都是 Gitea 的模板语法,如果您不完全理解这些组件,不建议您对它们进行修改。
|
|
||||||
|
|
||||||
### 添加链接和页签
|
|
||||||
|
|
||||||
如果您只是想添加额外的链接到顶部导航栏或额外的选项卡到存储库视图,您可以将它们放在您 `custom/templates/custom/` 目录下的 `extra_links.tmpl` 和 `extra_tabs.tmpl` 文件中。
|
|
||||||
|
|
||||||
举例说明:假设您需要在网站放置一个静态的“关于”页面,您只需将该页面放在您的
|
|
||||||
"custom/public/"目录下(比如 `custom/public/impressum.html`)并且将它与 `custom/templates/custom/extra_links.tmpl` 链接起来即可。
|
|
||||||
|
|
||||||
这个链接应当使用一个名为“item”的 class 来匹配当前样式,您可以使用 `{{AppSubUrl}}` 来获取 base URL:
|
|
||||||
`<a class="item" href="{{AppSubUrl}}/assets/impressum.html">Impressum</a>`
|
|
||||||
|
|
||||||
同理,您可以将页签添加到 `extra_tabs.tmpl` 中,使用同样的方式来添加页签。它的具体样式需要与
|
|
||||||
`templates/repo/header.tmpl` 中已有的其他选项卡的样式匹配
|
|
||||||
([source in GitHub](https://github.com/go-gitea/gitea/blob/main/templates/repo/header.tmpl))
|
|
||||||
|
|
||||||
### 页面的其他新增内容
|
|
||||||
|
|
||||||
除了 `extra_links.tmpl` 和 `extra_tabs.tmpl`,您可以在您的 `custom/templates/custom/` 目录中存放一些其他有用的模板,例如:
|
|
||||||
|
|
||||||
- `header.tmpl`,在 `<head>` 标记结束之前的模板,例如添加自定义CSS文件
|
|
||||||
- `body_outer_pre.tmpl`,在 `<body>` 标记开始处的模板
|
|
||||||
- `body_inner_pre.tmpl`,在顶部导航栏之前,但在主 container 内部的模板,例如添加一个 `<div class="full height">`
|
|
||||||
- `body_inner_post.tmpl`,在主 container 结束处的模板
|
|
||||||
- `body_outer_post.tmpl`,在底部 `<footer>` 元素之前.
|
|
||||||
- `footer.tmpl`,在 `<body>` 标签结束处的模板,可以在这里填写一些附加的 Javascript 脚本。
|
|
||||||
|
|
||||||
## 自定义 gitignores,labels, licenses, locales 以及 readmes
|
|
||||||
|
|
||||||
将自定义文件放在 `custom/options` 下相应子的文件夹中即可
|
|
||||||
|
|
||||||
## 更改 Gitea 外观
|
|
||||||
|
|
||||||
内置主题是“gitea-light”、“gitea-dark”和“gitea-auto”(自动适应操作系统设置)。
|
|
||||||
|
|
||||||
默认主题可以通过 `app.ini` 的 [ui](administration/config-cheat-sheet.md#ui-ui) 部分中的 `DEFAULT_THEME` 进行更改。
|
|
@ -1,86 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2019-10-15T10:10:00+05:00"
|
|
||||||
title: "Email setup"
|
|
||||||
slug: "email-setup"
|
|
||||||
sidebar_position: 12
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/email-setup
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Email setup"
|
|
||||||
sidebar_position: 12
|
|
||||||
identifier: "email-setup"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Email setup
|
|
||||||
|
|
||||||
Gitea has mailer functionality for sending transactional emails (such as registration confirmation). It can be configured to either use Sendmail (or compatible MTAs like Postfix and msmtp) or directly use SMTP server.
|
|
||||||
|
|
||||||
## Using Sendmail
|
|
||||||
|
|
||||||
Use `sendmail` command as mailer.
|
|
||||||
|
|
||||||
Note: For use in the official Gitea Docker image, please configure with the SMTP version (see the following section).
|
|
||||||
|
|
||||||
Note: For Internet-facing sites consult documentation of your MTA for instructions to send emails over TLS. Also set up SPF, DMARC, and DKIM DNS records to make emails sent be accepted as legitimate by various email providers.
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[mailer]
|
|
||||||
ENABLED = true
|
|
||||||
FROM = gitea@mydomain.com
|
|
||||||
PROTOCOL = sendmail
|
|
||||||
SENDMAIL_PATH = /usr/sbin/sendmail
|
|
||||||
SENDMAIL_ARGS = "--" ; most "sendmail" programs take options, "--" will prevent an email address being interpreted as an option.
|
|
||||||
```
|
|
||||||
|
|
||||||
## Using SMTP
|
|
||||||
|
|
||||||
Directly use SMTP server as relay. This option is useful if you don't want to set up MTA on your instance but you have an account at email provider.
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[mailer]
|
|
||||||
ENABLED = true
|
|
||||||
FROM = gitea@mydomain.com
|
|
||||||
PROTOCOL = smtps
|
|
||||||
SMTP_ADDR = mail.mydomain.com
|
|
||||||
SMTP_PORT = 587
|
|
||||||
USER = gitea@mydomain.com
|
|
||||||
PASSWD = `password`
|
|
||||||
```
|
|
||||||
|
|
||||||
Restart Gitea for the configuration changes to take effect.
|
|
||||||
|
|
||||||
To send a test email to validate the settings, go to Gitea > Site Administration > Configuration > SMTP Mailer Configuration.
|
|
||||||
|
|
||||||
For the full list of options check the [Config Cheat Sheet](administration/config-cheat-sheet.md)
|
|
||||||
|
|
||||||
Please note: authentication is only supported when the SMTP server communication is encrypted with TLS or `HOST=localhost`. TLS encryption can be through:
|
|
||||||
|
|
||||||
- STARTTLS (also known as Opportunistic TLS) via port 587. Initial connection is done over cleartext, but then be upgraded over TLS if the server supports it.
|
|
||||||
- SMTPS connection (SMTP over TLS) via the default port 465. Connection to the server use TLS from the beginning.
|
|
||||||
- Forced SMTPS connection with `PROTOCOL=smtps`. (These are both known as Implicit TLS.)
|
|
||||||
This is due to protections imposed by the Go internal libraries against STRIPTLS attacks.
|
|
||||||
|
|
||||||
Note that Implicit TLS is recommended by [RFC8314](https://tools.ietf.org/html/rfc8314#section-3) since 2018.
|
|
||||||
|
|
||||||
### Gmail
|
|
||||||
|
|
||||||
The following configuration should work with GMail's SMTP server:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[mailer]
|
|
||||||
ENABLED = true
|
|
||||||
HOST = smtp.gmail.com:465 ; Remove this line for Gitea >= 1.18.0
|
|
||||||
SMTP_ADDR = smtp.gmail.com
|
|
||||||
SMTP_PORT = 465
|
|
||||||
FROM = example.user@gmail.com
|
|
||||||
USER = example.user
|
|
||||||
PASSWD = `***`
|
|
||||||
PROTOCOL = smtps
|
|
||||||
```
|
|
||||||
|
|
||||||
Note that you'll need to create and use an [App password](https://support.google.com/accounts/answer/185833?hl=en) by enabling 2FA on your Google
|
|
||||||
account. You won't be able to use your Google account password directly.
|
|
@ -1,85 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2023-05-23T09:00:00+08:00"
|
|
||||||
title: "Email 设置"
|
|
||||||
slug: "email-setup"
|
|
||||||
sidebar_position: 12
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/email-setup
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Email 设置"
|
|
||||||
sidebar_position: 12
|
|
||||||
identifier: "email-setup"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Email 设置
|
|
||||||
|
|
||||||
Gitea 具有邮件功能,用于发送事务性邮件(例如注册确认邮件)。它可以配置为使用 Sendmail(或兼容的 MTA,例如 Postfix 和 msmtp)或直接使用 SMTP 服务器。
|
|
||||||
|
|
||||||
## 使用 Sendmail
|
|
||||||
|
|
||||||
使用 `sendmail` 命令作为邮件传输代理(mailer)。
|
|
||||||
|
|
||||||
注意:对于在官方Gitea Docker镜像中使用,请使用SMTP版本进行配置(请参考下一节)。
|
|
||||||
|
|
||||||
注意:对于面向互联网的网站,请查阅您的 MTA 文档以了解通过TLS发送邮件的说明。同时设置 SPF、DMARC 和 DKIM DNS 记录,以使发送的邮件被各个电子邮件提供商接受为合法邮件。
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[mailer]
|
|
||||||
ENABLED = true
|
|
||||||
FROM = gitea@mydomain.com
|
|
||||||
PROTOCOL = sendmail
|
|
||||||
SENDMAIL_PATH = /usr/sbin/sendmail
|
|
||||||
SENDMAIL_ARGS = "--" ; 大多数 "sendmail" 程序都接受选项,使用 "--" 将防止电子邮件地址被解释为选项。
|
|
||||||
```
|
|
||||||
|
|
||||||
## 使用 SMTP
|
|
||||||
|
|
||||||
直接使用 SMTP 服务器作为中继。如果您不想在实例上设置 MTA,但在电子邮件提供商那里有一个帐户,这个选项非常有用。
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[mailer]
|
|
||||||
ENABLED = true
|
|
||||||
FROM = gitea@mydomain.com
|
|
||||||
PROTOCOL = smtps
|
|
||||||
SMTP_ADDR = mail.mydomain.com
|
|
||||||
SMTP_PORT = 587
|
|
||||||
USER = gitea@mydomain.com
|
|
||||||
PASSWD = `password`
|
|
||||||
```
|
|
||||||
|
|
||||||
重启 Gitea 以使配置更改生效。
|
|
||||||
|
|
||||||
要发送测试邮件以验证设置,请转到 Gitea > 站点管理 > 配置 > SMTP 邮件配置。
|
|
||||||
|
|
||||||
有关所有选项的完整列表,请查看[配置速查表](administration/config-cheat-sheet.md)。
|
|
||||||
|
|
||||||
请注意:只有在使用 TLS 或 `HOST=localhost` 加密 SMTP 服务器通信时才支持身份验证。TLS 加密可以通过以下方式进行:
|
|
||||||
|
|
||||||
- 通过端口 587 的 STARTTLS(也称为 Opportunistic TLS)。初始连接是明文的,但如果服务器支持,则可以升级为 TLS。
|
|
||||||
- 通过默认端口 465 的 SMTPS 连接。连接到服务器从一开始就使用 TLS。
|
|
||||||
- 使用 `PROTOCOL=smtps` 进行强制的 SMTPS 连接。(这两种方式都被称为 Implicit TLS)
|
|
||||||
这是由于 Go 内部库对 STRIPTLS 攻击的保护机制。
|
|
||||||
|
|
||||||
请注意,自2018年起,[RFC8314](https://tools.ietf.org/html/rfc8314#section-3) 推荐使用 Implicit TLS。
|
|
||||||
|
|
||||||
### Gmail
|
|
||||||
|
|
||||||
以下配置应该适用于 Gmail 的 SMTP 服务器:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[mailer]
|
|
||||||
ENABLED = true
|
|
||||||
HOST = smtp.gmail.com:465 ; 对于 Gitea >= 1.18.0,删除此行
|
|
||||||
SMTP_ADDR = smtp.gmail.com
|
|
||||||
SMTP_PORT = 465
|
|
||||||
FROM = example.user@gmail.com
|
|
||||||
USER = example.user
|
|
||||||
PASSWD = `***`
|
|
||||||
PROTOCOL = smtps
|
|
||||||
```
|
|
||||||
|
|
||||||
请注意,您需要创建并使用一个 [应用密码](https://support.google.com/accounts/answer/185833?hl=en) 并在您的 Google 帐户上启用 2FA。您将无法直接使用您的 Google 帐户密码。
|
|
@ -1,59 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2017-04-08T11:34:00+02:00"
|
|
||||||
title: "Environment variables"
|
|
||||||
slug: "environment-variables"
|
|
||||||
sidebar_position: 10
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/environment-variables
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Environment variables"
|
|
||||||
sidebar_position: 10
|
|
||||||
identifier: "environment-variables"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Environment variables
|
|
||||||
|
|
||||||
This is an inventory of Gitea environment variables. They change Gitea behaviour.
|
|
||||||
|
|
||||||
Initialize them before Gitea command to be effective, for example:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
GITEA_CUSTOM=/home/gitea/custom ./gitea web
|
|
||||||
```
|
|
||||||
|
|
||||||
## From Go language
|
|
||||||
|
|
||||||
As Gitea is written in Go, it uses some variables that influence the behaviour of Go's runtime, such as:
|
|
||||||
|
|
||||||
- `GOMEMLIMIT`
|
|
||||||
- `GOGC`
|
|
||||||
- `GOMAXPROCS`
|
|
||||||
- `GODEBUG`
|
|
||||||
|
|
||||||
For documentation about each of the variables available, refer to the
|
|
||||||
[official Go documentation on runtime environment variables](https://pkg.go.dev/runtime#hdr-Environment_Variables).
|
|
||||||
|
|
||||||
## Gitea files
|
|
||||||
|
|
||||||
- `GITEA_WORK_DIR`: Absolute path of working directory.
|
|
||||||
- `GITEA_CUSTOM`: Gitea uses `WorkPath`/custom folder by default. Use this variable to change _custom_ directory.
|
|
||||||
|
|
||||||
## Operating system specifics
|
|
||||||
|
|
||||||
- `USER`: System user that Gitea will run as. Used for some repository access strings.
|
|
||||||
- `USERNAME`: if no `USER` found, Gitea will use `USERNAME`
|
|
||||||
- `HOME`: User home directory path. The `USERPROFILE` environment variable is used in Windows.
|
|
||||||
|
|
||||||
### Only on Windows
|
|
||||||
|
|
||||||
- `USERPROFILE`: User home directory path. If empty, uses `HOMEDRIVE` + `HOMEPATH`
|
|
||||||
- `HOMEDRIVE`: Main drive path used to access the home directory (C:)
|
|
||||||
- `HOMEPATH`: Home relative path in the given home drive path
|
|
||||||
|
|
||||||
## Miscellaneous
|
|
||||||
|
|
||||||
- `SKIP_MINWINSVC`: If set to 1, do not run as a service on Windows.
|
|
@ -1,57 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2017-04-08T11:34:00+02:00"
|
|
||||||
title: "环境变量清单"
|
|
||||||
slug: "environment-variables"
|
|
||||||
sidebar_position: 10
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/environment-variables
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "环境变量清单"
|
|
||||||
sidebar_position: 10
|
|
||||||
identifier: "environment-variables"
|
|
||||||
---
|
|
||||||
|
|
||||||
# 环境变量清单
|
|
||||||
|
|
||||||
这里是用来控制 Gitea 行为表现的的环境变量清单,您需要在执行如下 Gitea 启动命令前设置它们来确保配置生效:
|
|
||||||
|
|
||||||
```
|
|
||||||
GITEA_CUSTOM=/home/gitea/custom ./gitea web
|
|
||||||
```
|
|
||||||
|
|
||||||
## Go 的配置
|
|
||||||
|
|
||||||
因为 Gitea 使用 Go 语言编写,因此它使用了一些相关的 Go 的配置参数:
|
|
||||||
|
|
||||||
* `GOOS`
|
|
||||||
* `GOARCH`
|
|
||||||
* [`GOPATH`](https://go.dev/cmd/go/#hdr-GOPATH_environment_variable)
|
|
||||||
|
|
||||||
您可以在[官方文档](https://go.dev/cmd/go/#hdr-Environment_variables)中查阅这些配置参数的详细信息。
|
|
||||||
|
|
||||||
## Gitea 的文件目录
|
|
||||||
|
|
||||||
* `GITEA_WORK_DIR`:工作目录的绝对路径
|
|
||||||
* `GITEA_CUSTOM`:默认情况下 Gitea 使用默认目录 `GITEA_WORK_DIR`/custom,您可以使用这个参数来配置 *custom* 目录
|
|
||||||
* `GOGS_WORK_DIR`: 已废弃,请使用 `GITEA_WORK_DIR` 替代
|
|
||||||
* `GOGS_CUSTOM`: 已废弃,请使用 `GITEA_CUSTOM` 替代
|
|
||||||
|
|
||||||
## 操作系统配置
|
|
||||||
|
|
||||||
* `USER`:Gitea 运行时使用的系统用户,它将作为一些 repository 的访问地址的一部分
|
|
||||||
* `USERNAME`: 如果没有配置 `USER`, Gitea 将使用 `USERNAME`
|
|
||||||
* `HOME`: 用户的 home 目录,在 Windows 中会使用 `USERPROFILE` 环境变量
|
|
||||||
|
|
||||||
### 仅限于 Windows 的配置
|
|
||||||
|
|
||||||
* `USERPROFILE`: 用户的主目录,如果未配置则会使用 `HOMEDRIVE` + `HOMEPATH`
|
|
||||||
* `HOMEDRIVE`: 用于访问 home 目录的主驱动器路径(C盘)
|
|
||||||
* `HOMEPATH`:在指定主驱动器下的 home 目录相对路径
|
|
||||||
|
|
||||||
## Miscellaneous
|
|
||||||
|
|
||||||
* `SKIP_MINWINSVC`:如果设置为 1,在 Windows 上不会以 service 的形式运行。
|
|
@ -1,194 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2018-11-23:00:00+02:00"
|
|
||||||
title: "External renderers"
|
|
||||||
slug: "external-renderers"
|
|
||||||
sidebar_position: 60
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/external-renderers
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "External renderers"
|
|
||||||
sidebar_position: 60
|
|
||||||
identifier: "external-renderers"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Custom files rendering configuration
|
|
||||||
|
|
||||||
Gitea supports custom file renderings (i.e., Jupyter notebooks, asciidoc, etc.) through external binaries,
|
|
||||||
it is just a matter of:
|
|
||||||
|
|
||||||
- installing external binaries
|
|
||||||
- add some configuration to your `app.ini` file
|
|
||||||
- restart your Gitea instance
|
|
||||||
|
|
||||||
This supports rendering of whole files. If you want to render code blocks in markdown you would need to do something with javascript. See some examples on the [Customizing Gitea](administration/customizing-gitea.md) page.
|
|
||||||
|
|
||||||
## Installing external binaries
|
|
||||||
|
|
||||||
In order to get file rendering through external binaries, their associated packages must be installed.
|
|
||||||
If you're using a Docker image, your `Dockerfile` should contain something along this lines:
|
|
||||||
|
|
||||||
```docker
|
|
||||||
FROM gitea/gitea:@version@
|
|
||||||
[...]
|
|
||||||
|
|
||||||
COPY custom/app.ini /data/gitea/conf/app.ini
|
|
||||||
[...]
|
|
||||||
|
|
||||||
RUN apk --no-cache add asciidoctor freetype freetype-dev gcc g++ libpng libffi-dev py-pip python3-dev py3-pip py3-pyzmq
|
|
||||||
# install any other package you need for your external renderers
|
|
||||||
|
|
||||||
RUN pip3 install --upgrade pip
|
|
||||||
RUN pip3 install -U setuptools
|
|
||||||
RUN pip3 install jupyter docutils
|
|
||||||
# add above any other python package you may need to install
|
|
||||||
```
|
|
||||||
|
|
||||||
## `app.ini` file configuration
|
|
||||||
|
|
||||||
add one `[markup.XXXXX]` section per external renderer on your custom `app.ini`:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[markup.asciidoc]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .adoc,.asciidoc
|
|
||||||
RENDER_COMMAND = "asciidoctor -s -a showtitle --out-file=- -"
|
|
||||||
; Input is not a standard input but a file
|
|
||||||
IS_INPUT_FILE = false
|
|
||||||
|
|
||||||
[markup.jupyter]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .ipynb
|
|
||||||
RENDER_COMMAND = "jupyter nbconvert --stdin --stdout --to html --template basic"
|
|
||||||
IS_INPUT_FILE = false
|
|
||||||
|
|
||||||
[markup.restructuredtext]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .rst
|
|
||||||
RENDER_COMMAND = "timeout 30s pandoc +RTS -M512M -RTS -f rst"
|
|
||||||
IS_INPUT_FILE = false
|
|
||||||
```
|
|
||||||
|
|
||||||
If your external markup relies on additional classes and attributes on the generated HTML elements, you might need to enable custom sanitizer policies. Gitea uses the [`bluemonday`](https://godoc.org/github.com/microcosm-cc/bluemonday) package as our HTML sanitizer. The example below could be used to support server-side [KaTeX](https://katex.org/) rendering output from [`pandoc`](https://pandoc.org/).
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[markup.sanitizer.TeX]
|
|
||||||
; Pandoc renders TeX segments as <span>s with the "math" class, optionally
|
|
||||||
; with "inline" or "display" classes depending on context.
|
|
||||||
; - note this is different from the built-in math support in our markdown parser which uses <code>
|
|
||||||
ELEMENT = span
|
|
||||||
ALLOW_ATTR = class
|
|
||||||
REGEXP = ^\s*((math(\s+|$)|inline(\s+|$)|display(\s+|$)))+
|
|
||||||
|
|
||||||
[markup.markdown]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .md,.markdown
|
|
||||||
RENDER_COMMAND = pandoc -f markdown -t html --katex
|
|
||||||
```
|
|
||||||
|
|
||||||
You must define `ELEMENT` and `ALLOW_ATTR` in each section.
|
|
||||||
|
|
||||||
To define multiple entries, add a unique alphanumeric suffix (e.g., `[markup.sanitizer.1]` and `[markup.sanitizer.something]`).
|
|
||||||
|
|
||||||
To apply a sanitisation rules only for a specify external renderer they must use the renderer name, e.g. `[markup.sanitizer.asciidoc.rule-1]`, `[markup.sanitizer.<renderer>.rule-1]`.
|
|
||||||
|
|
||||||
**Note**: If the rule is defined above the renderer ini section or the name does not match a renderer it is applied to every renderer.
|
|
||||||
|
|
||||||
Once your configuration changes have been made, restart Gitea to have changes take effect.
|
|
||||||
|
|
||||||
**Note**: Prior to Gitea 1.12 there was a single `markup.sanitiser` section with keys that were redefined for multiple rules, however,
|
|
||||||
there were significant problems with this method of configuration necessitating configuration through multiple sections.
|
|
||||||
|
|
||||||
### Example: HTML
|
|
||||||
|
|
||||||
Render HTML files directly:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[markup.html]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .html,.htm
|
|
||||||
RENDER_COMMAND = cat
|
|
||||||
; Input is not a standard input but a file
|
|
||||||
IS_INPUT_FILE = true
|
|
||||||
|
|
||||||
[markup.sanitizer.html.1]
|
|
||||||
ELEMENT = div
|
|
||||||
ALLOW_ATTR = class
|
|
||||||
|
|
||||||
[markup.sanitizer.html.2]
|
|
||||||
ELEMENT = a
|
|
||||||
ALLOW_ATTR = class
|
|
||||||
```
|
|
||||||
|
|
||||||
### Example: Office DOCX
|
|
||||||
|
|
||||||
Display Office DOCX files with [`pandoc`](https://pandoc.org/):
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[markup.docx]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .docx
|
|
||||||
RENDER_COMMAND = "pandoc --from docx --to html --self-contained --template /path/to/basic.html"
|
|
||||||
|
|
||||||
[markup.sanitizer.docx.img]
|
|
||||||
ALLOW_DATA_URI_IMAGES = true
|
|
||||||
```
|
|
||||||
|
|
||||||
The template file has the following content:
|
|
||||||
|
|
||||||
```
|
|
||||||
$body$
|
|
||||||
```
|
|
||||||
|
|
||||||
### Example: Jupyter Notebook
|
|
||||||
|
|
||||||
Display Jupyter Notebook files with [`nbconvert`](https://github.com/jupyter/nbconvert):
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[markup.jupyter]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .ipynb
|
|
||||||
RENDER_COMMAND = "jupyter-nbconvert --stdin --stdout --to html --template basic"
|
|
||||||
|
|
||||||
[markup.sanitizer.jupyter.img]
|
|
||||||
ALLOW_DATA_URI_IMAGES = true
|
|
||||||
```
|
|
||||||
|
|
||||||
## Customizing CSS
|
|
||||||
|
|
||||||
The external renderer is specified in the .ini in the format `[markup.XXXXX]` and the HTML supplied by your external renderer will be wrapped in a `<div>` with classes `markup` and `XXXXX`. The `markup` class provides out of the box styling (as does `markdown` if `XXXXX` is `markdown`). Otherwise you can use these classes to specifically target the contents of your rendered HTML.
|
|
||||||
|
|
||||||
And so you could write some CSS:
|
|
||||||
|
|
||||||
```css
|
|
||||||
.markup.XXXXX html {
|
|
||||||
font-size: 100%;
|
|
||||||
overflow-y: scroll;
|
|
||||||
-webkit-text-size-adjust: 100%;
|
|
||||||
-ms-text-size-adjust: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markup.XXXXX body {
|
|
||||||
color: #444;
|
|
||||||
font-family: Georgia, Palatino, 'Palatino Linotype', Times, 'Times New Roman', serif;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 1.7;
|
|
||||||
padding: 1em;
|
|
||||||
margin: auto;
|
|
||||||
max-width: 42em;
|
|
||||||
background: #fefefe;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markup.XXXXX p {
|
|
||||||
color: orangered;
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Add your stylesheet to your custom directory e.g `custom/public/assets/css/my-style-XXXXX.css` and import it using a custom header file `custom/templates/custom/header.tmpl`:
|
|
||||||
|
|
||||||
```html
|
|
||||||
<link rel="stylesheet" href="{{AppSubUrl}}/assets/css/my-style-XXXXX.css" />
|
|
||||||
```
|
|
@ -1,203 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2023-05-23T09:00:00+08:00"
|
|
||||||
title: "外部渲染器"
|
|
||||||
slug: "external-renderers"
|
|
||||||
sidebar_position: 60
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/external-renderers
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "外部渲染器"
|
|
||||||
sidebar_position: 60
|
|
||||||
identifier: "external-renderers"
|
|
||||||
---
|
|
||||||
|
|
||||||
# 自定义文件渲染配置
|
|
||||||
|
|
||||||
Gitea 通过外部二进制文件支持自定义文件渲染(例如 Jupyter notebooks、asciidoc 等),只需要进行以下步骤:
|
|
||||||
|
|
||||||
- 安装外部二进制文件
|
|
||||||
- 在您的 `app.ini` 文件中添加一些配置
|
|
||||||
- 重新启动 Gitea 实例
|
|
||||||
|
|
||||||
此功能支持整个文件的渲染。如果您想要在 Markdown 中渲染代码块,您需要使用 JavaScript 进行一些操作。请参阅 [自定义 Gitea 配置](administration/customizing-gitea.md) 页面上的一些示例。
|
|
||||||
|
|
||||||
## 安装外部二进制文件
|
|
||||||
|
|
||||||
为了通过外部二进制文件进行文件渲染,必须安装它们的关联软件包。
|
|
||||||
如果您正在使用 Docker 镜像,则您的 `Dockerfile` 应该包含以下内容:
|
|
||||||
|
|
||||||
```docker
|
|
||||||
FROM gitea/gitea:@version@
|
|
||||||
[...]
|
|
||||||
|
|
||||||
COPY custom/app.ini /data/gitea/conf/app.ini
|
|
||||||
[...]
|
|
||||||
|
|
||||||
RUN apk --no-cache add asciidoctor freetype freetype-dev gcc g++ libpng libffi-dev py-pip python3-dev py3-pip py3-pyzmq
|
|
||||||
# 安装其他您需要的外部渲染器的软件包
|
|
||||||
|
|
||||||
RUN pip3 install --upgrade pip
|
|
||||||
RUN pip3 install -U setuptools
|
|
||||||
RUN pip3 install jupyter docutils
|
|
||||||
# 在上面添加您需要安装的任何其他 Python 软件包
|
|
||||||
```
|
|
||||||
|
|
||||||
## `app.ini` 文件配置
|
|
||||||
|
|
||||||
在您的自定义 `app.ini` 文件中为每个外部渲染器添加一个 `[markup.XXXXX]` 部分:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[markup.asciidoc]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .adoc,.asciidoc
|
|
||||||
RENDER_COMMAND = "asciidoctor -s -a showtitle --out-file=- -"
|
|
||||||
; 输入不是标准输入而是文件
|
|
||||||
IS_INPUT_FILE = false
|
|
||||||
|
|
||||||
[markup.jupyter]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .ipynb
|
|
||||||
RENDER_COMMAND = "jupyter nbconvert --stdin --stdout --to html --template basic"
|
|
||||||
IS_INPUT_FILE = false
|
|
||||||
|
|
||||||
[markup.restructuredtext]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .rst
|
|
||||||
RENDER_COMMAND = "timeout 30s pandoc +RTS -M512M -RTS -f rst"
|
|
||||||
IS_INPUT_FILE = false
|
|
||||||
```
|
|
||||||
|
|
||||||
如果您的外部标记语言依赖于在生成的 HTML 元素上的额外类和属性,您可能需要启用自定义的清理策略。Gitea 使用 [`bluemonday`](https://godoc.org/github.com/microcosm-cc/bluemonday) 包作为我们的 HTML 清理器。下面的示例可以用于支持从 [`pandoc`](https://pandoc.org/) 输出的服务器端 [KaTeX](https://katex.org/) 渲染结果。
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[markup.sanitizer.TeX]
|
|
||||||
; Pandoc 渲染 TeX 段落为带有 "math" 类的 <span> 元素,根据上下文可能还带有 "inline" 或 "display" 类。
|
|
||||||
; - 请注意,这与我们的 Markdown 解析器中内置的数学支持不同,后者使用 <code> 元素。
|
|
||||||
ELEMENT = span
|
|
||||||
ALLOW_ATTR = class
|
|
||||||
REGEXP = ^\s*((math(\s+|$)|inline(\s+|$)|display(\s+|$)))+
|
|
||||||
|
|
||||||
[markup.markdown]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .md,.markdown
|
|
||||||
RENDER_COMMAND = pandoc -f markdown -t html --katex
|
|
||||||
```
|
|
||||||
|
|
||||||
您必须在每个部分中定义 `ELEMENT` 和 `ALLOW_ATTR`。
|
|
||||||
|
|
||||||
要定义多个条目,请添加唯一的字母数字后缀(例如,`[markup.sanitizer.1]` 和 `[markup.sanitizer.something]`)。
|
|
||||||
|
|
||||||
要仅为特定的外部渲染器应用清理规则,它们必须使用渲染器名称,例如 `[markup.sanitizer.asciidoc.rule-1]`、`[markup.sanitizer.<renderer>.rule-1]`。
|
|
||||||
|
|
||||||
**注意**:如果规则在渲染器 ini 部分之前定义,或者名称与渲染器不匹配,它将应用于所有渲染器。
|
|
||||||
|
|
||||||
完成配置更改后,请重新启动 Gitea 以使更改生效。
|
|
||||||
|
|
||||||
**注意**:在 Gitea 1.12 之前,存在一个名为 `markup.sanitiser` 的单个部分,其中的键被重新定义为多个规则,但是,这种配置方法存在重大问题,需要通过多个部分进行配置。
|
|
||||||
|
|
||||||
### 示例:HTML
|
|
||||||
|
|
||||||
直接渲染 HTML 文件:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[markup.html]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .html,.htm
|
|
||||||
RENDER_COMMAND = cat
|
|
||||||
; 输入不是标准输入,而是文件
|
|
||||||
IS_INPUT_FILE = true
|
|
||||||
|
|
||||||
[markup.sanitizer.html.1]
|
|
||||||
ELEMENT = div
|
|
||||||
ALLOW_ATTR = class
|
|
||||||
|
|
||||||
[markup.sanitizer.html.2]
|
|
||||||
ELEMENT = a
|
|
||||||
ALLOW_ATTR = class
|
|
||||||
```
|
|
||||||
|
|
||||||
请注意:此示例中的配置将允许渲染 HTML 文件,并使用 `cat` 命令将文件内容输出为 HTML。此外,配置中的两个清理规则将允许 `<div>` 和 `<a>` 元素使用 `class` 属性。
|
|
||||||
|
|
||||||
在进行配置更改后,请重新启动 Gitea 以使更改生效。
|
|
||||||
|
|
||||||
### 示例:Office DOCX
|
|
||||||
|
|
||||||
使用 [`pandoc`](https://pandoc.org/) 显示 Office DOCX 文件:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[markup.docx]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .docx
|
|
||||||
RENDER_COMMAND = "pandoc --from docx --to html --self-contained --template /path/to/basic.html"
|
|
||||||
|
|
||||||
[markup.sanitizer.docx.img]
|
|
||||||
ALLOW_DATA_URI_IMAGES = true
|
|
||||||
```
|
|
||||||
|
|
||||||
在此示例中,配置将允许显示 Office DOCX 文件,并使用 `pandoc` 命令将文件转换为 HTML 格式。同时,清理规则中的 `ALLOW_DATA_URI_IMAGES` 设置为 `true`,允许使用 Data URI 格式的图片。
|
|
||||||
|
|
||||||
模板文件的内容如下:
|
|
||||||
|
|
||||||
```
|
|
||||||
$body$
|
|
||||||
```
|
|
||||||
|
|
||||||
### 示例:Jupyter Notebook
|
|
||||||
|
|
||||||
使用 [`nbconvert`](https://github.com/jupyter/nbconvert) 显示 Jupyter Notebook 文件:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[markup.jupyter]
|
|
||||||
ENABLED = true
|
|
||||||
FILE_EXTENSIONS = .ipynb
|
|
||||||
RENDER_COMMAND = "jupyter-nbconvert --stdin --stdout --to html --template basic"
|
|
||||||
|
|
||||||
[markup.sanitizer.jupyter.img]
|
|
||||||
ALLOW_DATA_URI_IMAGES = true
|
|
||||||
```
|
|
||||||
|
|
||||||
在此示例中,配置将允许显示 Jupyter Notebook 文件,并使用 `nbconvert` 命令将文件转换为 HTML 格式。同样,清理规则中的 `ALLOW_DATA_URI_IMAGES` 设置为 `true`,允许使用 Data URI 格式的图片。
|
|
||||||
|
|
||||||
在进行配置更改后,请重新启动 Gitea 以使更改生效。
|
|
||||||
|
|
||||||
## 自定义 CSS
|
|
||||||
|
|
||||||
在 `.ini` 文件中,可以使用 `[markup.XXXXX]` 的格式指定外部渲染器,并且由外部渲染器生成的 HTML 将被包装在一个带有 `markup` 和 `XXXXX` 类的 `<div>` 中。`markup` 类提供了预定义的样式(如果 `XXXXX` 是 `markdown`,则使用 `markdown` 类)。否则,您可以使用这些类来针对渲染的 HTML 内容进行定制样式。
|
|
||||||
|
|
||||||
因此,您可以编写一些 CSS 样式:
|
|
||||||
|
|
||||||
```css
|
|
||||||
.markup.XXXXX html {
|
|
||||||
font-size: 100%;
|
|
||||||
overflow-y: scroll;
|
|
||||||
-webkit-text-size-adjust: 100%;
|
|
||||||
-ms-text-size-adjust: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markup.XXXXX body {
|
|
||||||
color: #444;
|
|
||||||
font-family: Georgia, Palatino, 'Palatino Linotype', Times, 'Times New Roman', serif;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 1.7;
|
|
||||||
padding: 1em;
|
|
||||||
margin: auto;
|
|
||||||
max-width: 42em;
|
|
||||||
background: #fefefe;
|
|
||||||
}
|
|
||||||
|
|
||||||
.markup.XXXXX p {
|
|
||||||
color: orangered;
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
将您的样式表添加到自定义目录中,例如 `custom/public/assets/css/my-style-XXXXX.css`,并使用自定义的头文件 `custom/templates/custom/header.tmpl` 进行导入:
|
|
||||||
|
|
||||||
```html
|
|
||||||
<link rel="stylesheet" href="{{AppSubUrl}}/assets/css/my-style-XXXXX.css" />
|
|
||||||
```
|
|
||||||
|
|
||||||
通过以上步骤,您可以将自定义的 CSS 样式应用到特定的外部渲染器,使其具有所需的样式效果。
|
|
@ -1,127 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2018-05-11T11:00:00+02:00"
|
|
||||||
title: "Fail2ban Setup "
|
|
||||||
slug: "fail2ban-setup"
|
|
||||||
sidebar_position: 16
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/fail2ban-setup
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Fail2ban setup"
|
|
||||||
sidebar_position: 16
|
|
||||||
identifier: "fail2ban-setup"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Fail2ban setup to block users after failed login attempts
|
|
||||||
|
|
||||||
**Remember that fail2ban is powerful and can cause lots of issues if you do it incorrectly, so make
|
|
||||||
sure to test this before relying on it so you don't lock yourself out.**
|
|
||||||
|
|
||||||
Gitea returns an HTTP 200 for bad logins in the web logs, but if you have logging options on in
|
|
||||||
`app.ini`, then you should be able to go off of `log/gitea.log`, which gives you something like this
|
|
||||||
on a bad authentication from the web or CLI using SSH or HTTP respectively:
|
|
||||||
|
|
||||||
```log
|
|
||||||
2018/04/26 18:15:54 [I] Failed authentication attempt for user from xxx.xxx.xxx.xxx
|
|
||||||
```
|
|
||||||
|
|
||||||
```log
|
|
||||||
2020/10/15 16:05:09 modules/ssh/ssh.go:143:publicKeyHandler() [W] Failed authentication attempt from xxx.xxx.xxx.xxx
|
|
||||||
```
|
|
||||||
|
|
||||||
(DEPRECATED: This may be a false positive as the user may still go on to correctly authenticate.)
|
|
||||||
|
|
||||||
```log
|
|
||||||
2020/10/15 16:05:09 modules/ssh/ssh.go:155:publicKeyHandler() [W] Failed authentication attempt from xxx.xxx.xxx.xxx
|
|
||||||
```
|
|
||||||
|
|
||||||
(DEPRECATED: This may be a false positive as the user may still go on to correctly authenticate.)
|
|
||||||
|
|
||||||
```log
|
|
||||||
2020/10/15 16:05:09 modules/ssh/ssh.go:198:publicKeyHandler() [W] Failed authentication attempt from xxx.xxx.xxx.xxx
|
|
||||||
```
|
|
||||||
|
|
||||||
(DEPRECATED: This may be a false positive as the user may still go on to correctly authenticate.)
|
|
||||||
|
|
||||||
```log
|
|
||||||
2020/10/15 16:05:09 modules/ssh/ssh.go:213:publicKeyHandler() [W] Failed authentication attempt from xxx.xxx.xxx.xxx
|
|
||||||
```
|
|
||||||
|
|
||||||
(DEPRECATED: This may be a false positive as the user may still go on to correctly authenticate.)
|
|
||||||
|
|
||||||
```log
|
|
||||||
2020/10/15 16:05:09 modules/ssh/ssh.go:227:publicKeyHandler() [W] Failed authentication attempt from xxx.xxx.xxx.xxx
|
|
||||||
```
|
|
||||||
|
|
||||||
(DEPRECATED: This may be a false positive as the user may still go on to correctly authenticate.)
|
|
||||||
|
|
||||||
```log
|
|
||||||
2020/10/15 16:05:09 modules/ssh/ssh.go:249:sshConnectionFailed() [W] Failed authentication attempt from xxx.xxx.xxx.xxx
|
|
||||||
```
|
|
||||||
|
|
||||||
(From 1.15 this new message will available and doesn't have any of the false positive results that above messages from publicKeyHandler do. This will only be logged if the user has completely failed authentication.)
|
|
||||||
|
|
||||||
```log
|
|
||||||
2020/10/15 16:08:44 ...s/context/context.go:204:HandleText() [E] invalid credentials from xxx.xxx.xxx.xxx
|
|
||||||
```
|
|
||||||
|
|
||||||
Add our filter in `/etc/fail2ban/filter.d/gitea.conf`:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
# gitea.conf
|
|
||||||
[Definition]
|
|
||||||
failregex = .*(Failed authentication attempt|invalid credentials|Attempted access of unknown user).* from <HOST>
|
|
||||||
ignoreregex =
|
|
||||||
```
|
|
||||||
|
|
||||||
Add our jail in `/etc/fail2ban/jail.d/gitea.conf`:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[gitea]
|
|
||||||
enabled = true
|
|
||||||
filter = gitea
|
|
||||||
logpath = /var/lib/gitea/log/gitea.log
|
|
||||||
maxretry = 10
|
|
||||||
findtime = 3600
|
|
||||||
bantime = 900
|
|
||||||
action = iptables-allports
|
|
||||||
```
|
|
||||||
|
|
||||||
If you're using Docker, you'll also need to add an additional jail to handle the **FORWARD**
|
|
||||||
chain in **iptables**. Configure it in `/etc/fail2ban/jail.d/gitea-docker.conf`:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[gitea-docker]
|
|
||||||
enabled = true
|
|
||||||
filter = gitea
|
|
||||||
logpath = /var/lib/gitea/log/gitea.log
|
|
||||||
maxretry = 10
|
|
||||||
findtime = 3600
|
|
||||||
bantime = 900
|
|
||||||
action = iptables-allports[chain="FORWARD"]
|
|
||||||
```
|
|
||||||
|
|
||||||
Then simply run `service fail2ban restart` to apply your changes. You can check to see if
|
|
||||||
fail2ban has accepted your configuration using `service fail2ban status`.
|
|
||||||
|
|
||||||
Make sure and read up on fail2ban and configure it to your needs, this bans someone
|
|
||||||
for **15 minutes** (from all ports) when they fail authentication 10 times in an hour.
|
|
||||||
|
|
||||||
If you run Gitea behind a reverse proxy with Nginx (for example with Docker), you need to add
|
|
||||||
this to your Nginx configuration so that IPs don't show up as 127.0.0.1:
|
|
||||||
|
|
||||||
```
|
|
||||||
proxy_set_header X-Real-IP $remote_addr;
|
|
||||||
```
|
|
||||||
|
|
||||||
The security options in `app.ini` need to be adjusted to allow the interpretation of the headers
|
|
||||||
as well as the list of IP addresses and networks that describe trusted proxy servers
|
|
||||||
(See the [configuration cheat sheet](administration/config-cheat-sheet.md#security-security) for more information).
|
|
||||||
|
|
||||||
```
|
|
||||||
REVERSE_PROXY_LIMIT = 1
|
|
||||||
REVERSE_PROXY_TRUSTED_PROXIES = 127.0.0.1/8 ; 172.17.0.0/16 for the docker default network
|
|
||||||
```
|
|
@ -1,94 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2022-08-01T00:00:00+00:00"
|
|
||||||
title: "设置 Fail2ban"
|
|
||||||
slug: "fail2ban-setup"
|
|
||||||
sidebar_position: 16
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/fail2ban-setup
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "设置 Fail2ban"
|
|
||||||
sidebar_position: 16
|
|
||||||
identifier: "fail2ban-setup"
|
|
||||||
---
|
|
||||||
|
|
||||||
# 使用 Fail2ban 阻止攻击者的暴力登录
|
|
||||||
|
|
||||||
**Fail2ban 检查客户端登录日志,将多次登录失败的客户端识别为攻击者并在一段时间内阻止其访问服务。如果你的实例是公开的,这一点尤其重要。请管理员仔细设置 fail2ban,错误的配置将导致防火墙阻止你访问自己的服务器。**
|
|
||||||
|
|
||||||
Gitea 会在日志文件 `log/gitea.log` 中记录登录失败的 CLI、SSH 或 HTTP 客户端 IP 地址,而你需要将 Gitea 的日志输出模式从默认的 `console` 更改为 `file`。这表示将日志输出到文件,使得 fail2ban 可以定期扫描日志内容。
|
|
||||||
|
|
||||||
当用户的身份验证失败时,日志中会记录此类信息:
|
|
||||||
|
|
||||||
```log
|
|
||||||
2018/04/26 18:15:54 [I] Failed authentication attempt for user from xxx.xxx.xxx.xxx
|
|
||||||
```
|
|
||||||
|
|
||||||
```log
|
|
||||||
2020/10/15 16:08:44 [E] invalid credentials from xxx.xxx.xxx.xxx
|
|
||||||
```
|
|
||||||
|
|
||||||
## 设置 Fail2ban
|
|
||||||
|
|
||||||
添加日志过滤器规则到配置文件 `/etc/fail2ban/filter.d/gitea.conf`:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[Definition]
|
|
||||||
failregex = .*(Failed authentication attempt|invalid credentials|Attempted access of unknown user).* from <HOST>
|
|
||||||
ignoreregex =
|
|
||||||
```
|
|
||||||
|
|
||||||
添加监狱规则到配置文件 `/etc/fail2ban/jail.d/gitea.conf`:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[gitea]
|
|
||||||
enabled = true
|
|
||||||
filter = gitea
|
|
||||||
logpath = /var/lib/gitea/log/gitea.log
|
|
||||||
maxretry = 10
|
|
||||||
findtime = 3600
|
|
||||||
bantime = 900
|
|
||||||
action = iptables-allports
|
|
||||||
```
|
|
||||||
|
|
||||||
如果你的 Gitea 实例运行在 Docker 容器中,并且直接将容器端口暴露到外部网络,
|
|
||||||
你还需要添加 `chain="FORWARD"` 到监狱规则配置文件 `/etc/fail2ban/jail.d/gitea-docker.conf`
|
|
||||||
以适应 Docker 的网络转发规则。但如果你在容器的宿主机上使用 Nginx 反向代理连接到 Gitea 则无需这样配置。
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[gitea-docker]
|
|
||||||
enabled = true
|
|
||||||
filter = gitea
|
|
||||||
logpath = /var/lib/gitea/log/gitea.log
|
|
||||||
maxretry = 10
|
|
||||||
findtime = 3600
|
|
||||||
bantime = 900
|
|
||||||
action = iptables-allports[chain="FORWARD"]
|
|
||||||
```
|
|
||||||
|
|
||||||
最后,运行 `systemctl restart fail2ban` 即可应用更改。现在,你可以使用 `systemctl status fail2ban` 检查 fail2ban 运行状态。
|
|
||||||
|
|
||||||
上述规则规定客户端在 1 小时内,如果登录失败的次数达到 10 次,则通过 iptables 锁定该客户端 IP 地址 15 分钟。
|
|
||||||
|
|
||||||
## 设置反向代理
|
|
||||||
|
|
||||||
如果你使用 Nginx 反向代理到 Gitea 实例,你还需要设置 Nginx 的 HTTP 头部值 `X-Real-IP` 将真实的客户端 IP 地址传递给 Gitea。否则 Gitea 程序会将客户端地址错误解析为反向代理服务器的地址,例如回环地址 `127.0.0.1`。
|
|
||||||
|
|
||||||
```
|
|
||||||
proxy_set_header X-Real-IP $remote_addr;
|
|
||||||
```
|
|
||||||
|
|
||||||
额外注意,在 Gitea 的配置文件 `app.ini` 中存在下列默认值:
|
|
||||||
|
|
||||||
```
|
|
||||||
REVERSE_PROXY_LIMIT = 1
|
|
||||||
REVERSE_PROXY_TRUSTED_PROXIES = 127.0.0.0/8,::1/128
|
|
||||||
```
|
|
||||||
|
|
||||||
`REVERSE_PROXY_LIMIT` 限制反向代理服务器的层数,设置为 `0` 表示不使用这些标头。
|
|
||||||
`REVERSE_PROXY_TRUSTED_PROXIES` 表示受信任的反向代理服务器网络地址,
|
|
||||||
经过该网络地址转发来的流量会经过解析 `X-Real-IP` 头部得到真实客户端地址。
|
|
||||||
(参考 [configuration cheat sheet](administration/config-cheat-sheet.md#security-security))
|
|
@ -1,32 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2019-10-06T08:00:00+05:00"
|
|
||||||
title: "Git LFS setup"
|
|
||||||
slug: "git-lfs-setup"
|
|
||||||
sidebar_position: 12
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/git-lfs-setup
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Git LFS setup"
|
|
||||||
sidebar_position: 12
|
|
||||||
identifier: "git-lfs-setup"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Git Large File Storage setup
|
|
||||||
|
|
||||||
To use Gitea's built-in LFS support, you must update the `app.ini` file:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[server]
|
|
||||||
; Enables git-lfs support. true or false, default is false.
|
|
||||||
LFS_START_SERVER = true
|
|
||||||
|
|
||||||
[lfs]
|
|
||||||
; Where your lfs files reside, default is data/lfs.
|
|
||||||
PATH = /home/gitea/data/lfs
|
|
||||||
```
|
|
||||||
|
|
||||||
**Note**: LFS server support needs at least Git v2.1.2 installed on the server
|
|
@ -1,32 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2023-05-23T09:00:00+08:00"
|
|
||||||
title: "Git LFS 设置"
|
|
||||||
slug: "git-lfs-setup"
|
|
||||||
sidebar_position: 12
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/git-lfs-setup
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Git LFS 设置"
|
|
||||||
sidebar_position: 12
|
|
||||||
identifier: "git-lfs-setup"
|
|
||||||
---
|
|
||||||
|
|
||||||
# 配置 Git 大文件存储(Large File Storage,LFS)
|
|
||||||
|
|
||||||
要使用 Gitea 内置的 LFS 支持,您需要更新 `app.ini` 文件:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[server]
|
|
||||||
; 启用 git-lfs 支持。true 或 false,默认为 false。
|
|
||||||
LFS_START_SERVER = true
|
|
||||||
|
|
||||||
[lfs]
|
|
||||||
; 存放 LFS 文件的路径,默认为 data/lfs。
|
|
||||||
PATH = /home/gitea/data/lfs
|
|
||||||
```
|
|
||||||
|
|
||||||
**注意**:LFS 服务器支持需要服务器上安装 Git v2.1.2 以上版本。
|
|
@ -1,100 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2018-06-02T11:00:00+02:00"
|
|
||||||
title: "HTTPS setup"
|
|
||||||
slug: "https-setup"
|
|
||||||
sidebar_position: 12
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/https-setup
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "HTTPS setup"
|
|
||||||
sidebar_position: 12
|
|
||||||
identifier: "https-setup"
|
|
||||||
---
|
|
||||||
|
|
||||||
# HTTPS setup to encrypt connections to Gitea
|
|
||||||
|
|
||||||
## Using the built-in server
|
|
||||||
|
|
||||||
Before you enable HTTPS, make sure that you have valid SSL/TLS certificates.
|
|
||||||
You could use self-generated certificates for evaluation and testing. Please run `gitea cert --host [HOST]` to generate a self signed certificate.
|
|
||||||
|
|
||||||
If you are using Apache or nginx on the server, it's recommended to check the [reverse proxy guide](administration/reverse-proxies.md).
|
|
||||||
|
|
||||||
To use Gitea's built-in HTTPS support, you must change your `app.ini` file:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[server]
|
|
||||||
PROTOCOL = https
|
|
||||||
ROOT_URL = https://git.example.com:3000/
|
|
||||||
HTTP_PORT = 3000
|
|
||||||
CERT_FILE = cert.pem
|
|
||||||
KEY_FILE = key.pem
|
|
||||||
```
|
|
||||||
|
|
||||||
Note that if your certificate is signed by a third party certificate authority (i.e. not self-signed), then cert.pem should contain the certificate chain. The server certificate must be the first entry in cert.pem, followed by the intermediaries in order (if any). The root certificate does not have to be included because the connecting client must already have it in order to establish the trust relationship.
|
|
||||||
To learn more about the config values, please checkout the [Config Cheat Sheet](administration/config-cheat-sheet.md#server-server).
|
|
||||||
|
|
||||||
For the `CERT_FILE` or `KEY_FILE` field, the file path is relative to the `GITEA_CUSTOM` environment variable when it is a relative path. It can be an absolute path as well.
|
|
||||||
|
|
||||||
### Setting up HTTP redirection
|
|
||||||
|
|
||||||
The Gitea server is only able to listen to one port; to redirect HTTP requests to the HTTPS port, you will need to enable the HTTP redirection service:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[server]
|
|
||||||
REDIRECT_OTHER_PORT = true
|
|
||||||
; Port the redirection service should listen on
|
|
||||||
PORT_TO_REDIRECT = 3080
|
|
||||||
```
|
|
||||||
|
|
||||||
If you are using Docker, make sure that this port is configured in your `docker-compose.yml` file.
|
|
||||||
|
|
||||||
## Using ACME (Default: Let's Encrypt)
|
|
||||||
|
|
||||||
[ACME](https://tools.ietf.org/html/rfc8555) is a Certificate Authority standard protocol that allows you to automatically request and renew SSL/TLS certificates. [Let's Encrypt](https://letsencrypt.org/) is a free publicly trusted Certificate Authority server using this standard. Only `HTTP-01` and `TLS-ALPN-01` challenges are implemented. In order for ACME challenges to pass and verify your domain ownership, external traffic to the gitea domain on port `80` (`HTTP-01`) or port `443` (`TLS-ALPN-01`) has to be served by the gitea instance. Setting up [HTTP redirection](#setting-up-http-redirection) and port-forwards might be needed for external traffic to route correctly. Normal traffic to port `80` will otherwise be automatically redirected to HTTPS. **You must consent** to the ACME provider's terms of service (default Let's Encrypt's [terms of service](https://letsencrypt.org/documents/LE-SA-v1.2-November-15-2017.pdf)).
|
|
||||||
|
|
||||||
Minimum setup using the default Let's Encrypt:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[server]
|
|
||||||
PROTOCOL=https
|
|
||||||
DOMAIN=git.example.com
|
|
||||||
ENABLE_ACME=true
|
|
||||||
ACME_ACCEPTTOS=true
|
|
||||||
ACME_DIRECTORY=https
|
|
||||||
;; Email can be omitted here and provided manually at first run, after which it is cached
|
|
||||||
ACME_EMAIL=email@example.com
|
|
||||||
```
|
|
||||||
|
|
||||||
Minimum setup using a [smallstep CA](https://github.com/smallstep/certificates), refer to [their tutorial](https://smallstep.com/docs/tutorials/acme-challenge) for more information.
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[server]
|
|
||||||
PROTOCOL=https
|
|
||||||
DOMAIN=git.example.com
|
|
||||||
ENABLE_ACME=true
|
|
||||||
ACME_ACCEPTTOS=true
|
|
||||||
ACME_URL=https://ca.example.com/acme/acme/directory
|
|
||||||
;; Can be omitted if using the system's trust is preferred
|
|
||||||
;ACME_CA_ROOT=/path/to/root_ca.crt
|
|
||||||
ACME_DIRECTORY=https
|
|
||||||
ACME_EMAIL=email@example.com
|
|
||||||
```
|
|
||||||
|
|
||||||
To learn more about the config values, please checkout the [Config Cheat Sheet](administration/config-cheat-sheet.md#server-server).
|
|
||||||
|
|
||||||
## Using a reverse proxy
|
|
||||||
|
|
||||||
Setup up your reverse proxy as shown in the [reverse proxy guide](administration/reverse-proxies.md).
|
|
||||||
|
|
||||||
After that, enable HTTPS by following one of these guides:
|
|
||||||
|
|
||||||
- [nginx](https://nginx.org/en/docs/http/configuring_https_servers.html)
|
|
||||||
- [apache2/httpd](https://httpd.apache.org/docs/2.4/ssl/ssl_howto.html)
|
|
||||||
- [caddy](https://caddyserver.com/docs/tls)
|
|
||||||
|
|
||||||
Note: Enabling HTTPS only at the proxy level is referred as [TLS Termination Proxy](https://en.wikipedia.org/wiki/TLS_termination_proxy). The proxy server accepts incoming TLS connections, decrypts the contents, and passes the now unencrypted contents to Gitea. This is normally fine as long as both the proxy and Gitea instances are either on the same machine, or on different machines within private network (with the proxy is exposed to outside network). If your Gitea instance is separated from your proxy over a public network, or if you want full end-to-end encryption, you can also [enable HTTPS support directly in Gitea using built-in server](#using-the-built-in-server) and forward the connections over HTTPS instead.
|
|
@ -1,97 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2023-04-09T11:00:00+02:00"
|
|
||||||
title: "HTTPS配置"
|
|
||||||
slug: "https-setup"
|
|
||||||
sidebar_position: 12
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "HTTPS setup"
|
|
||||||
sidebar_position: 12
|
|
||||||
identifier: "https-setup"
|
|
||||||
---
|
|
||||||
|
|
||||||
# HTTPS setup to encrypt connections to Gitea
|
|
||||||
|
|
||||||
## 使用内置服务器
|
|
||||||
|
|
||||||
在启用HTTPS之前,确保您拥有有效的SSL/TLS证书。
|
|
||||||
建议在测试和评估情况下使用自签名证书,请运行 `gitea cert --host [HOST]` 以生成自签名证书
|
|
||||||
|
|
||||||
如果您在服务器上使用阿帕奇(Apache)或Nginx,建议参考 [反向代理指南](administration/reverse-proxies.md)。
|
|
||||||
|
|
||||||
要使用Gitea内置HTTPS支持,您必须编辑`app.ini`文件。
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[server]
|
|
||||||
PROTOCOL = https
|
|
||||||
ROOT_URL = https://git.example.com:3000/
|
|
||||||
HTTP_PORT = 3000
|
|
||||||
CERT_FILE = cert.pem
|
|
||||||
KEY_FILE = key.pem
|
|
||||||
```
|
|
||||||
|
|
||||||
请注意,如果您的证书由第三方证书颁发机构签名(即不是自签名的),则 cert.pem 应包含证书链。服务器证书必须是 cert.pem 中的第一个条目,后跟中介(如果有)。不必包含根证书,因为连接客户端必须已经拥有根证书才能建立信任关系。要了解有关配置值的更多信息,请查看 [配置备忘单](administration/config-cheat-sheet#server-server)。
|
|
||||||
|
|
||||||
对于“CERT_FILE”或“KEY_FILE”字段,当文件路径是相对路径时,文件路径相对于“GITEA_CUSTOM”环境变量。它也可以是绝对路径。
|
|
||||||
|
|
||||||
### 设置HTTP重定向
|
|
||||||
|
|
||||||
Gitea服务器仅支持监听一个端口;要重定向HTTP请求致HTTPS端口,您需要启用HTTP重定向服务:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[server]
|
|
||||||
REDIRECT_OTHER_PORT = true
|
|
||||||
; Port the redirection service should listen on
|
|
||||||
PORT_TO_REDIRECT = 3080
|
|
||||||
```
|
|
||||||
|
|
||||||
如果您使用Docker,确保端口已配置在 `docker-compose.yml` 文件
|
|
||||||
|
|
||||||
## 使用 ACME (默认: Let's Encrypt)
|
|
||||||
|
|
||||||
[ACME](https://tools.ietf.org/html/rfc8555) 是一种证书颁发机构标准协议,允许您自动请求和续订 SSL/TLS 证书。[Let`s Encrypt](https://letsencrypt.org/) 是使用此标准的免费公开信任的证书颁发机构服务器。仅实施“HTTP-01”和“TLS-ALPN-01”挑战。为了使 ACME 质询通过并验证您的域所有权,“80”端口(“HTTP-01”)或“443”端口(“TLS-ALPN-01”)上 gitea 域的外部流量必须由 gitea 实例提供服务。可能需要设置 [HTTP 重定向](#设置http重定向) 和端口转发才能正确路由外部流量。否则,到端口“80”的正常流量将自动重定向到 HTTPS。**您必须同意**ACME提供商的服务条款(默认为Let's Encrypt的 [服务条款](https://letsencrypt.org/documents/LE-SA-v1.2-2017年11月15日.pdf)。
|
|
||||||
|
|
||||||
使用默认 Let's Encrypt 的最小配置如下:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[server]
|
|
||||||
PROTOCOL=https
|
|
||||||
DOMAIN=git.example.com
|
|
||||||
ENABLE_ACME=true
|
|
||||||
ACME_ACCEPTTOS=true
|
|
||||||
ACME_DIRECTORY=https
|
|
||||||
;; Email can be omitted here and provided manually at first run, after which it is cached
|
|
||||||
ACME_EMAIL=email@example.com
|
|
||||||
```
|
|
||||||
|
|
||||||
小型配置请使用 [smallstep CA](https://github.com/smallstep/certificates), 点击 [教程](https://smallstep.com/docs/tutorials/acme-challenge) 了解更多信息。
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[server]
|
|
||||||
PROTOCOL=https
|
|
||||||
DOMAIN=git.example.com
|
|
||||||
ENABLE_ACME=true
|
|
||||||
ACME_ACCEPTTOS=true
|
|
||||||
ACME_URL=https://ca.example.com/acme/acme/directory
|
|
||||||
;; Can be omitted if using the system's trust is preferred
|
|
||||||
;ACME_CA_ROOT=/path/to/root_ca.crt
|
|
||||||
ACME_DIRECTORY=https
|
|
||||||
ACME_EMAIL=email@example.com
|
|
||||||
```
|
|
||||||
|
|
||||||
要了解关于配置, 请访问 [配置备忘单](administration/config-cheat-sheet.md#server-server)获取更多信息
|
|
||||||
|
|
||||||
## 使用反向代理服务器
|
|
||||||
|
|
||||||
按照 [reverse proxy guide](administration/reverse-proxies.md) 的规则设置你的反向代理服务器
|
|
||||||
|
|
||||||
然后,按照下面的向导启用 HTTPS:
|
|
||||||
|
|
||||||
- [nginx](https://nginx.org/en/docs/http/configuring_https_servers.html)
|
|
||||||
- [apache2/httpd](https://httpd.apache.org/docs/2.4/ssl/ssl_howto.html)
|
|
||||||
- [caddy](https://caddyserver.com/docs/tls)
|
|
||||||
|
|
||||||
注意:仅在代理层启用 HTTPS 被称为 [TLS 终止代理](https://en.wikipedia.org/wiki/TLS_termination_proxy)。代理服务器接受传入的 TLS 连接,解密内容,然后将现在未加密的内容传递给 Gitea。只要代理和 Gitea 实例在同一台计算机上或在私有网络中的不同计算机上(代理暴露给外部网络),这通常是可以接受的。如果您的 Gitea 实例与代理隔离在公共网络上,或者如果您想要全端到端的加密,您还可以直接在 Gitea 中 [启用内置服务器的 HTTPS 支持](#使用内置服务器),并将连接转发到 HTTPS 上。
|
|
@ -1,294 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2019-04-02T17:06:00+01:00"
|
|
||||||
title: "Logging Configuration"
|
|
||||||
slug: "logging-config"
|
|
||||||
sidebar_position: 40
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/logging-configuration
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Logging Configuration"
|
|
||||||
sidebar_position: 40
|
|
||||||
identifier: "logging-config"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Logging Configuration
|
|
||||||
|
|
||||||
The logging configuration of Gitea mainly consists of 3 types of components:
|
|
||||||
|
|
||||||
- The `[log]` section for general configuration
|
|
||||||
- `[log.<mode-name>]` sections for the configuration of different log writers to output logs, aka: "writer mode", the mode name is also used as "writer name".
|
|
||||||
- The `[log]` section can also contain sub-logger configurations following the key schema `logger.<logger-name>.<CONFIG-KEY>`
|
|
||||||
|
|
||||||
There is a fully functional log output by default, so it is not necessary to define one.
|
|
||||||
|
|
||||||
## Collecting Logs for Help
|
|
||||||
|
|
||||||
To collect logs for help and issue report, see [Support Options](help/support.md).
|
|
||||||
|
|
||||||
## The `[log]` section
|
|
||||||
|
|
||||||
Configuration of logging facilities in Gitea happen in the `[log]` section and its subsections.
|
|
||||||
|
|
||||||
In the top level `[log]` section the following configurations can be placed:
|
|
||||||
|
|
||||||
- `ROOT_PATH`: (Default: **%(GITEA_WORK_DIR)/log**): Base path for log files
|
|
||||||
- `MODE`: (Default: **console**) List of log outputs to use for the Default logger.
|
|
||||||
- `LEVEL`: (Default: **Info**) Least severe log events to persist, case-insensitive. Possible values are: `Trace`, `Debug`, `Info`, `Warn`, `Error`, `Fatal`.
|
|
||||||
- `STACKTRACE_LEVEL`: (Default: **None**) For this and more severe events the stacktrace will be printed upon getting logged.
|
|
||||||
|
|
||||||
And it can contain the following sub-loggers:
|
|
||||||
|
|
||||||
- `logger.router.MODE`: (Default: **,**): List of log outputs to use for the Router logger.
|
|
||||||
- `logger.access.MODE`: (Default: **_empty_**) List of log outputs to use for the Access logger. By default, the access logger is disabled.
|
|
||||||
- `logger.xorm.MODE`: (Default: **,**) List of log outputs to use for the XORM logger.
|
|
||||||
|
|
||||||
Setting a comma (`,`) to sub-logger's mode means making it use the default global `MODE`.
|
|
||||||
|
|
||||||
## Quick samples
|
|
||||||
|
|
||||||
### Default (empty) Configuration
|
|
||||||
|
|
||||||
The empty configuration is equivalent to default:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[log]
|
|
||||||
ROOT_PATH = %(GITEA_WORK_DIR)/log
|
|
||||||
MODE = console
|
|
||||||
LEVEL = Info
|
|
||||||
STACKTRACE_LEVEL = None
|
|
||||||
logger.router.MODE = ,
|
|
||||||
logger.xorm.MODE = ,
|
|
||||||
logger.access.MODE =
|
|
||||||
|
|
||||||
; this is the config options of "console" mode (used by MODE=console above)
|
|
||||||
[log.console]
|
|
||||||
MODE = console
|
|
||||||
FLAGS = stdflags
|
|
||||||
PREFIX =
|
|
||||||
COLORIZE = true
|
|
||||||
```
|
|
||||||
|
|
||||||
This is equivalent to sending all logs to the console, with default Golang log being sent to the console log too.
|
|
||||||
|
|
||||||
This is only a sample, and it is the default, do not need to write it into your configuration file.
|
|
||||||
|
|
||||||
### Disable Router logs and record some access logs to file
|
|
||||||
|
|
||||||
The Router logger is disabled, the access logs (>=Warn) goes into `access.log`:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[log]
|
|
||||||
logger.router.MODE =
|
|
||||||
logger.access.MODE = access-file
|
|
||||||
|
|
||||||
[log.access-file]
|
|
||||||
MODE = file
|
|
||||||
LEVEL = Warn
|
|
||||||
FILE_NAME = access.log
|
|
||||||
```
|
|
||||||
|
|
||||||
### Set different log levels for different modes
|
|
||||||
|
|
||||||
Default logs (>=Warn) goes into `gitea.log`, while Error logs goes into `file-error.log`:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[log]
|
|
||||||
LEVEL = Warn
|
|
||||||
MODE = file, file-error
|
|
||||||
|
|
||||||
; by default, the "file" mode will record logs to %(log.ROOT_PATH)/gitea.log, so we don't need to set it
|
|
||||||
; [log.file]
|
|
||||||
; by default, the MODE (actually it's the output writer of this logger) is taken from the section name, so we don't need to set it either
|
|
||||||
; MODE = file
|
|
||||||
|
|
||||||
[log.file-error]
|
|
||||||
MODE = file
|
|
||||||
LEVEL = Error
|
|
||||||
FILE_NAME = file-error.log
|
|
||||||
```
|
|
||||||
|
|
||||||
## Log outputs (mode and writer)
|
|
||||||
|
|
||||||
Gitea provides the following log output writers:
|
|
||||||
|
|
||||||
- `console` - Log to `stdout` (or `stderr` if it is set in the config)
|
|
||||||
- `file` - Log to a file
|
|
||||||
- `conn` - Log to a socket (network or unix)
|
|
||||||
|
|
||||||
### Common configuration
|
|
||||||
|
|
||||||
Certain configuration is common to all modes of log output:
|
|
||||||
|
|
||||||
- `MODE` is the mode of the log output writer. It will default to the mode name in the ini section. Thus `[log.console]` will default to `MODE = console`.
|
|
||||||
- `LEVEL` is the lowest level that this output will log.
|
|
||||||
- `STACKTRACE_LEVEL` is the lowest level that this output will print a stacktrace.
|
|
||||||
- `COLORIZE` will default to `true` for `console` as described, otherwise it will default to `false`.
|
|
||||||
|
|
||||||
#### `EXPRESSION`
|
|
||||||
|
|
||||||
`EXPRESSION` represents a regular expression that log events must match to be logged by the output writer.
|
|
||||||
Either the log message, (with colors removed), must match or the `longfilename:linenumber:functionname` must match.
|
|
||||||
NB: the whole message or string doesn't need to completely match.
|
|
||||||
|
|
||||||
Please note this expression will be run in the writer's goroutine but not the logging event goroutine.
|
|
||||||
|
|
||||||
#### `FLAGS`
|
|
||||||
|
|
||||||
`FLAGS` represents the preceding logging context information that is
|
|
||||||
printed before each message. It is a comma-separated string set. The order of values does not matter.
|
|
||||||
|
|
||||||
It defaults to `stdflags` (= `date,time,medfile,shortfuncname,levelinitial`)
|
|
||||||
|
|
||||||
Possible values are:
|
|
||||||
|
|
||||||
- `none` or `,` - No flags.
|
|
||||||
- `date` - the date in the local time zone: `2009/01/23`.
|
|
||||||
- `time` - the time in the local time zone: `01:23:23`.
|
|
||||||
- `microseconds` - microsecond resolution: `01:23:23.123123`. Assumes time.
|
|
||||||
- `longfile` - full file name and line number: `/a/b/c/d.go:23`.
|
|
||||||
- `shortfile` - final file name element and line number: `d.go:23`.
|
|
||||||
- `funcname` - function name of the caller: `runtime.Caller()`.
|
|
||||||
- `shortfuncname` - last part of the function name. Overrides `funcname`.
|
|
||||||
- `utc` - if date or time is set, use UTC rather than the local time zone.
|
|
||||||
- `levelinitial` - initial character of the provided level in brackets eg. `[I]` for info.
|
|
||||||
- `level` - level in brackets `[INFO]`.
|
|
||||||
- `gopid` - the Goroutine-PID of the context.
|
|
||||||
- `medfile` - last 20 characters of the filename - equivalent to `shortfile,longfile`.
|
|
||||||
- `stdflags` - equivalent to `date,time,medfile,shortfuncname,levelinitial`.
|
|
||||||
|
|
||||||
### Console mode
|
|
||||||
|
|
||||||
In this mode the logger will forward log messages to the stdout and
|
|
||||||
stderr streams attached to the Gitea process.
|
|
||||||
|
|
||||||
For loggers in console mode, `COLORIZE` will default to `true` if not
|
|
||||||
on windows, or the Windows terminal can be set into ANSI mode or is a
|
|
||||||
cygwin or Msys pipe.
|
|
||||||
|
|
||||||
Settings:
|
|
||||||
|
|
||||||
- `STDERR`: **false**: Whether the logger should print to `stderr` instead of `stdout`.
|
|
||||||
|
|
||||||
### File mode
|
|
||||||
|
|
||||||
In this mode the logger will save log messages to a file.
|
|
||||||
|
|
||||||
Settings:
|
|
||||||
|
|
||||||
- `FILE_NAME`: The file to write the log events to, relative to `ROOT_PATH`, Default to `%(ROOT_PATH)/gitea.log`. Exception: access log will default to `%(ROOT_PATH)/access.log`.
|
|
||||||
- `MAX_SIZE_SHIFT`: **28**: Maximum size shift of a single file. 28 represents 256Mb. For details see below.
|
|
||||||
- `LOG_ROTATE` **true**: Whether to rotate the log files. TODO: if false, will it delete instead on daily rotate, or do nothing?.
|
|
||||||
- `DAILY_ROTATE`: **true**: Whether to rotate logs daily.
|
|
||||||
- `MAX_DAYS`: **7**: Delete rotated log files after this number of days.
|
|
||||||
- `COMPRESS`: **true**: Whether to compress old log files by default with gzip.
|
|
||||||
- `COMPRESSION_LEVEL`: **-1**: Compression level. For details see below.
|
|
||||||
|
|
||||||
`MAX_SIZE_SHIFT` defines the maximum size of a file by left shifting 1 the given number of times (`1 << x`).
|
|
||||||
The exact behavior at the time of v1.17.3 can be seen [here](https://github.com/go-gitea/gitea/blob/v1.17.3/modules/setting/log.go#L185).
|
|
||||||
|
|
||||||
The useful values of `COMPRESSION_LEVEL` are from 1 to (and including) 9, where higher numbers mean better compression.
|
|
||||||
Beware that better compression might come with higher resource usage.
|
|
||||||
Must be preceded with a `-` sign.
|
|
||||||
|
|
||||||
### Conn mode
|
|
||||||
|
|
||||||
In this mode the logger will send log messages over a network socket.
|
|
||||||
|
|
||||||
Settings:
|
|
||||||
|
|
||||||
- `ADDR`: **:7020**: Sets the address to connect to.
|
|
||||||
- `PROTOCOL`: **tcp**: Set the protocol, either "tcp", "unix" or "udp".
|
|
||||||
- `RECONNECT`: **false**: Try to reconnect when connection is lost.
|
|
||||||
- `RECONNECT_ON_MSG`: **false**: Reconnect host for every single message.
|
|
||||||
|
|
||||||
### The "Router" logger
|
|
||||||
|
|
||||||
The Router logger logs the following message types when Gitea's route handlers work:
|
|
||||||
|
|
||||||
- `started` messages will be logged at TRACE level
|
|
||||||
- `polling`/`completed` routers will be logged at INFO. Exception: "/assets" static resource requests are also logged at TRACE.
|
|
||||||
- `slow` routers will be logged at WARN
|
|
||||||
- `failed` routers will be logged at WARN
|
|
||||||
|
|
||||||
### The "XORM" logger
|
|
||||||
|
|
||||||
To make XORM outputs SQL logs, the `LOG_SQL` in `[database]` section should also be set to `true`.
|
|
||||||
|
|
||||||
### The "Access" logger
|
|
||||||
|
|
||||||
The Access logger is a new logger since Gitea 1.9. It provides a NCSA
|
|
||||||
Common Log compliant log format. It's highly configurable but caution
|
|
||||||
should be taken when changing its template. The main benefit of this
|
|
||||||
logger is that Gitea can now log accesses in a standard log format so
|
|
||||||
standard tools may be used.
|
|
||||||
|
|
||||||
You can enable this logger using `logger.access.MODE = ...`.
|
|
||||||
|
|
||||||
If desired the format of the Access logger can be changed by changing
|
|
||||||
the value of the `ACCESS_LOG_TEMPLATE`.
|
|
||||||
|
|
||||||
Please note, the access logger will log at `INFO` level, setting the
|
|
||||||
`LEVEL` of this logger to `WARN` or above will result in no access logs.
|
|
||||||
|
|
||||||
#### The ACCESS_LOG_TEMPLATE
|
|
||||||
|
|
||||||
This value represents a go template. Its default value is
|
|
||||||
|
|
||||||
```tmpl
|
|
||||||
{{.Ctx.RemoteHost}} - {{.Identity}} {{.Start.Format "[02/Jan/2006:15:04:05 -0700]" }} "{{.Ctx.Req.Method}} {{.Ctx.Req.URL.RequestURI}} {{.Ctx.Req.Proto}}" {{.ResponseWriter.Status}} {{.ResponseWriter.Size}} "{{.Ctx.Req.Referer}}" "{{.Ctx.Req.UserAgent}}"`
|
|
||||||
```
|
|
||||||
|
|
||||||
The template is passed following options:
|
|
||||||
|
|
||||||
- `Ctx` is the `context.Context`
|
|
||||||
- `Identity` is the `SignedUserName` or `"-"` if the user is not logged in
|
|
||||||
- `Start` is the start time of the request
|
|
||||||
- `ResponseWriter` is the `http.ResponseWriter`
|
|
||||||
|
|
||||||
Caution must be taken when changing this template as it runs outside of
|
|
||||||
the standard panic recovery trap. The template should also be as simple
|
|
||||||
as it runs for every request.
|
|
||||||
|
|
||||||
## Releasing-and-Reopening, Pausing and Resuming logging
|
|
||||||
|
|
||||||
If you are running on Unix you may wish to release-and-reopen logs in order to use `logrotate` or other tools.
|
|
||||||
It is possible force Gitea to release and reopen it's logging files and connections by sending `SIGUSR1` to the
|
|
||||||
running process, or running `gitea manager logging release-and-reopen`.
|
|
||||||
|
|
||||||
Alternatively, you may wish to pause and resume logging - this can be accomplished through the use of the
|
|
||||||
`gitea manager logging pause` and `gitea manager logging resume` commands. Please note that whilst logging
|
|
||||||
is paused log events below INFO level will not be stored and only a limited number of events will be stored.
|
|
||||||
Logging may block, albeit temporarily, slowing Gitea considerably whilst paused - therefore it is
|
|
||||||
recommended that pausing only done for a very short period of time.
|
|
||||||
|
|
||||||
## Adding and removing logging whilst Gitea is running
|
|
||||||
|
|
||||||
It is possible to add and remove logging whilst Gitea is running using the `gitea manager logging add` and `remove` subcommands.
|
|
||||||
This functionality can only adjust running log systems and cannot be used to start the access or router loggers if they
|
|
||||||
were not already initialized. If you wish to start these systems you are advised to adjust the app.ini and (gracefully) restart
|
|
||||||
the Gitea service.
|
|
||||||
|
|
||||||
The main intention of these commands is to easily add a temporary logger to investigate problems on running systems where a restart
|
|
||||||
may cause the issue to disappear.
|
|
||||||
|
|
||||||
## Using `logrotate` instead of built-in log rotation
|
|
||||||
|
|
||||||
Gitea includes built-in log rotation, which should be enough for most deployments. However, if you instead want to use the `logrotate` utility:
|
|
||||||
|
|
||||||
- Disable built-in log rotation by setting `LOG_ROTATE` to `false` in your `app.ini`.
|
|
||||||
- Install `logrotate`.
|
|
||||||
- Configure `logrotate` to match your deployment requirements, see `man 8 logrotate` for configuration syntax details.
|
|
||||||
In the `postrotate/endscript` block send Gitea a `USR1` signal via `kill -USR1` or `kill -10` to the `gitea` process itself,
|
|
||||||
or run `gitea manager logging release-and-reopen` (with the appropriate environment).
|
|
||||||
Ensure that your configurations apply to all files emitted by Gitea loggers as described in the above sections.
|
|
||||||
- Always do `logrotate /etc/logrotate.conf --debug` to test your configurations.
|
|
||||||
- If you are using docker and are running from outside the container you can use
|
|
||||||
`docker exec -u $OS_USER $CONTAINER_NAME sh -c 'gitea manager logging release-and-reopen'`
|
|
||||||
or `docker exec $CONTAINER_NAME sh -c '/bin/s6-svc -1 /etc/s6/gitea/'` or send `USR1` directly to the Gitea process itself.
|
|
||||||
|
|
||||||
The next `logrotate` jobs will include your configurations, so no restart is needed.
|
|
||||||
You can also immediately reload `logrotate` with `logrotate /etc/logrotate.conf --force`.
|
|
@ -1,272 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2023-05-23T09:00:00+08:00"
|
|
||||||
title: "日志配置"
|
|
||||||
slug: "logging-config"
|
|
||||||
sidebar_position: 40
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/logging-configuration
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "日志配置"
|
|
||||||
sidebar_position: 40
|
|
||||||
identifier: "logging-config"
|
|
||||||
---
|
|
||||||
|
|
||||||
# 日志配置
|
|
||||||
|
|
||||||
Gitea 的日志配置主要由以下三种类型的组件组成:
|
|
||||||
|
|
||||||
- `[log]` 部分用于一般配置
|
|
||||||
- `[log.<mode-name>]` 部分用于配置不同的日志输出方式,也称为 "writer mode",模式名称同时也作为 "writer name"
|
|
||||||
- `[log]` 部分还可以包含遵循 `logger.<logger-name>.<CONFIG-KEY>` 模式的子日志记录器的配置
|
|
||||||
|
|
||||||
默认情况下,已经有一个完全功能的日志输出,因此不需要重新定义。
|
|
||||||
|
|
||||||
## 收集日志以获取帮助
|
|
||||||
|
|
||||||
要收集日志以获取帮助和报告问题,请参阅 [需要帮助](help/support.md)。
|
|
||||||
|
|
||||||
## `[log]` 部分
|
|
||||||
|
|
||||||
在 Gitea 中,日志设施的配置在 `[log]` 部分及其子部分。
|
|
||||||
|
|
||||||
在顶层的 `[log]` 部分,可以放置以下配置项:
|
|
||||||
|
|
||||||
- `ROOT_PATH`:(默认值:**%(GITEA_WORK_DIR)/log**):日志文件的基本路径。
|
|
||||||
- `MODE`:(默认值:**console**):要用于默认日志记录器的日志输出列表。
|
|
||||||
- `LEVEL`:(默认值:**Info**):要持久化的最严重的日志事件,不区分大小写。可能的值为:`Trace`、`Debug`、`Info`、`Warn`、`Error`、`Fatal`。
|
|
||||||
- `STACKTRACE_LEVEL`:(默认值:**None**):对于此类及更严重的事件,将在记录时打印堆栈跟踪。
|
|
||||||
|
|
||||||
它还可以包含以下子日志记录器:
|
|
||||||
|
|
||||||
- `logger.router.MODE`:(默认值:**,**):用于路由器日志记录器的日志输出列表。
|
|
||||||
- `logger.access.MODE`:(默认值:**_empty_**):用于访问日志记录器的日志输出列表。默认情况下,访问日志记录器被禁用。
|
|
||||||
- `logger.xorm.MODE`:(默认值:**,**):用于 XORM 日志记录器的日志输出列表。
|
|
||||||
|
|
||||||
将子日志记录器的模式设置为逗号(`,`)表示使用默认的全局 `MODE`。
|
|
||||||
|
|
||||||
## 快速示例
|
|
||||||
|
|
||||||
### 默认(空)配置
|
|
||||||
|
|
||||||
空配置等同于默认配置:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[log]
|
|
||||||
ROOT_PATH = %(GITEA_WORK_DIR)/log
|
|
||||||
MODE = console
|
|
||||||
LEVEL = Info
|
|
||||||
STACKTRACE_LEVEL = None
|
|
||||||
logger.router.MODE = ,
|
|
||||||
logger.xorm.MODE = ,
|
|
||||||
logger.access.MODE =
|
|
||||||
|
|
||||||
; 这是“控制台”模式的配置选项(由上面的 MODE=console 使用)
|
|
||||||
[log.console]
|
|
||||||
MODE = console
|
|
||||||
FLAGS = stdflags
|
|
||||||
PREFIX =
|
|
||||||
COLORIZE = true
|
|
||||||
```
|
|
||||||
|
|
||||||
这等同于将所有日志发送到控制台,并将默认的 Golang 日志也发送到控制台日志中。
|
|
||||||
|
|
||||||
这只是一个示例,默认情况下不需要将其写入配置文件中。
|
|
||||||
|
|
||||||
### 禁用路由日志并将一些访问日志记录到文件中
|
|
||||||
|
|
||||||
禁用路由日志,将访问日志(>=Warn)记录到 `access.log` 中:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[log]
|
|
||||||
logger.router.MODE =
|
|
||||||
logger.access.MODE = access-file
|
|
||||||
|
|
||||||
[log.access-file]
|
|
||||||
MODE = file
|
|
||||||
LEVEL = Warn
|
|
||||||
FILE_NAME = access.log
|
|
||||||
```
|
|
||||||
|
|
||||||
### 为不同的模式设置不同的日志级别
|
|
||||||
|
|
||||||
将默认日志(>=Warn)记录到 `gitea.log` 中,将错误日志记录到 `file-error.log` 中:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[log]
|
|
||||||
LEVEL = Warn
|
|
||||||
MODE = file, file-error
|
|
||||||
|
|
||||||
; 默认情况下,"file" 模式会将日志记录到 %(log.ROOT_PATH)/gitea.log,因此我们不需要设置它
|
|
||||||
; [log.file]
|
|
||||||
|
|
||||||
[log.file-error]
|
|
||||||
LEVEL = Error
|
|
||||||
FILE_NAME = file-error.log
|
|
||||||
```
|
|
||||||
|
|
||||||
## 日志输出(模式和写入器)
|
|
||||||
|
|
||||||
Gitea 提供以下日志写入器:
|
|
||||||
|
|
||||||
- `console` - 输出日志到 `stdout`(或 `stderr`,如果已在配置中设置)
|
|
||||||
- `file` - 输出日志到文件
|
|
||||||
- `conn` - 输出日志到套接字(网络或 Unix 套接字)
|
|
||||||
|
|
||||||
### 公共配置
|
|
||||||
|
|
||||||
某些配置适用于所有日志输出模式:
|
|
||||||
|
|
||||||
- `MODE` 是日志输出写入器的模式。它将默认为 ini 部分的模式名称。因此,`[log.console]` 将默认为 `MODE = console`。
|
|
||||||
- `LEVEL` 是此输出将记录的最低日志级别。
|
|
||||||
- `STACKTRACE_LEVEL` 是此输出将打印堆栈跟踪的最低日志级别。
|
|
||||||
- `COLORIZE` 对于 `console`,默认为 `true`,否则默认为 `false`。
|
|
||||||
|
|
||||||
#### `EXPRESSION`
|
|
||||||
|
|
||||||
`EXPRESSION` 表示日志事件必须匹配才能被输出写入器记录的正则表达式。
|
|
||||||
日志消息(去除颜色)或 `longfilename:linenumber:functionname` 必须匹配其中之一。
|
|
||||||
注意:整个消息或字符串不需要完全匹配。
|
|
||||||
|
|
||||||
请注意,此表达式将在写入器的 goroutine 中运行,而不是在日志事件的 goroutine 中运行。
|
|
||||||
|
|
||||||
#### `FLAGS`
|
|
||||||
|
|
||||||
`FLAGS` 表示在每条消息之前打印的前置日志上下文信息。
|
|
||||||
它是一个逗号分隔的字符串集。值的顺序无关紧要。
|
|
||||||
|
|
||||||
默认值为 `stdflags`(= `date,time,medfile,shortfuncname,levelinitial`)。
|
|
||||||
|
|
||||||
可能的值为:
|
|
||||||
|
|
||||||
- `none` 或 `,` - 无标志。
|
|
||||||
- `date` - 当地时区的日期:`2009/01/23`。
|
|
||||||
- `time` - 当地时区的时间:`01:23:23`。
|
|
||||||
- `microseconds` - 微秒精度:`01:23:23.123123`。假定有时间。
|
|
||||||
- `longfile` - 完整的文件名和行号:`/a/b/c/d.go:23`。
|
|
||||||
- `shortfile` - 文件名的最后一个部分和行号:`d.go:23`。
|
|
||||||
- `funcname` - 调用者的函数名:`runtime.Caller()`。
|
|
||||||
- `shortfuncname` - 函数名的最后一部分。覆盖 `funcname`。
|
|
||||||
- `utc` - 如果设置了日期或时间,则使用 UTC 而不是本地时区。
|
|
||||||
- `levelinitial` - 提供的级别的初始字符,放在方括号内,例如 `[I]` 表示 info。
|
|
||||||
- `level` - 在方括号内的级别,例如 `[INFO]`。
|
|
||||||
- `gopid` - 上下文的 Goroutine-PID。
|
|
||||||
- `medfile` - 文件名的最后 20 个字符 - 相当于 `shortfile,longfile`。
|
|
||||||
- `stdflags` - 相当于 `date,time,medfile,shortfuncname,levelinitial`。
|
|
||||||
|
|
||||||
### Console 模式
|
|
||||||
|
|
||||||
在此模式下,日志记录器将将日志消息转发到 Gitea 进程附加的 stdout 和 stderr 流。
|
|
||||||
|
|
||||||
对于 console 模式的日志记录器,如果不在 Windows 上,或者 Windows 终端可以设置为 ANSI 模式,或者是 cygwin 或 Msys 管道,则 `COLORIZE` 默认为 `true`。
|
|
||||||
|
|
||||||
设置:
|
|
||||||
|
|
||||||
- `STDERR`:**false**:日志记录器是否应将日志打印到 `stderr` 而不是 `stdout`。
|
|
||||||
|
|
||||||
### File 模式
|
|
||||||
|
|
||||||
在此模式下,日志记录器将将日志消息保存到文件中。
|
|
||||||
|
|
||||||
设置:
|
|
||||||
|
|
||||||
- `FILE_NAME`:要将日志事件写入的文件,相对于 `ROOT_PATH`,默认为 `%(ROOT_PATH)/gitea.log`。异常情况:访问日志默认为 `%(ROOT_PATH)/access.log`。
|
|
||||||
- `MAX_SIZE_SHIFT`:**28**:单个文件的最大大小位移。28 表示 256Mb。详细信息见下文。
|
|
||||||
- `LOG_ROTATE` **true**:是否轮转日志文件。
|
|
||||||
- `DAILY_ROTATE`:**true**:是否每天旋转日志。
|
|
||||||
- `MAX_DAYS`:**7**:在此天数之后删除旋转的日志文件。
|
|
||||||
- `COMPRESS`:**true**:默认情况下是否使用 gzip 压缩旧的日志文件。
|
|
||||||
- `COMPRESSION_LEVEL`:**-1**:压缩级别。详细信息见下文。
|
|
||||||
|
|
||||||
`MAX_SIZE_SHIFT` 通过将给定次数左移 1 (`1 << x`) 来定义文件的最大大小。
|
|
||||||
在 v1.17.3 版本时的确切行为可以在[这里](https://github.com/go-gitea/gitea/blob/v1.17.3/modules/setting/log.go#L185)中查看。
|
|
||||||
|
|
||||||
`COMPRESSION_LEVEL` 的有用值范围从 1 到(包括)9,其中较高的数字表示更好的压缩。
|
|
||||||
请注意,更好的压缩可能会带来更高的资源使用。
|
|
||||||
必须在前面加上 `-` 符号。
|
|
||||||
|
|
||||||
### Conn 模式
|
|
||||||
|
|
||||||
在此模式下,日志记录器将通过网络套接字发送日志消息。
|
|
||||||
|
|
||||||
设置:
|
|
||||||
|
|
||||||
- `ADDR`:**:7020**:设置要连接的地址。
|
|
||||||
- `PROTOCOL`:**tcp**:设置协议,可以是 "tcp"、"unix" 或 "udp"。
|
|
||||||
- `RECONNECT`:**false**:在连接丢失时尝试重新连接。
|
|
||||||
- `RECONNECT_ON_MSG`:**false**:为每条消息重新连接主机。
|
|
||||||
|
|
||||||
### "Router" 日志记录器
|
|
||||||
|
|
||||||
当 Gitea 的路由处理程序工作时,Router 日志记录器记录以下消息类型:
|
|
||||||
|
|
||||||
- `started` 消息将以 TRACE 级别记录
|
|
||||||
- `polling`/`completed` 路由将以 INFO 级别记录。异常情况:"/assets" 静态资源请求也会以 TRACE 级别记录。
|
|
||||||
- `slow` 路由将以 WARN 级别记录
|
|
||||||
- `failed` 路由将以 WARN 级别记录
|
|
||||||
|
|
||||||
### "XORM" 日志记录器
|
|
||||||
|
|
||||||
为了使 XORM 输出 SQL 日志,还应将 `[database]` 部分中的 `LOG_SQL` 设置为 `true`。
|
|
||||||
|
|
||||||
### "Access" 日志记录器
|
|
||||||
|
|
||||||
"Access" 日志记录器是自 Gitea 1.9 版本以来的新日志记录器。它提供了符合 NCSA Common Log 标准的日志格式。虽然它具有高度可配置性,但在更改其模板时应谨慎。此日志记录器的主要好处是,Gitea 现在可以使用标准日志格式记录访问日志,因此可以使用标准工具进行分析。
|
|
||||||
|
|
||||||
您可以通过使用 `logger.access.MODE = ...` 来启用此日志记录器。
|
|
||||||
|
|
||||||
如果需要,可以通过更改 `ACCESS_LOG_TEMPLATE` 的值来更改 "Access" 日志记录器的格式。
|
|
||||||
|
|
||||||
请注意,访问日志记录器将以 `INFO` 级别记录,将此日志记录器的 `LEVEL` 设置为 `WARN` 或更高级别将导致不记录访问日志。
|
|
||||||
|
|
||||||
#### ACCESS_LOG_TEMPLATE
|
|
||||||
|
|
||||||
此值表示一个 Go 模板。其默认值为
|
|
||||||
|
|
||||||
```tmpl
|
|
||||||
{{.Ctx.RemoteHost}} - {{.Identity}} {{.Start.Format "[02/Jan/2006:15:04:05 -0700]" }} "{{.Ctx.Req.Method}} {{.Ctx.Req.URL.RequestURI}} {{.Ctx.Req.Proto}}" {{.ResponseWriter.Status}} {{.ResponseWriter.Size}} "{{.Ctx.Req.Referer}}" "{{.Ctx.Req.UserAgent}}"`
|
|
||||||
```
|
|
||||||
|
|
||||||
模板接收以下选项:
|
|
||||||
|
|
||||||
- `Ctx` 是 `context.Context`
|
|
||||||
- `Identity` 是 `SignedUserName`,如果用户未登录,则为 "-"
|
|
||||||
- `Start` 是请求的开始时间
|
|
||||||
- `ResponseWriter` 是 `http.ResponseWriter`
|
|
||||||
|
|
||||||
更改此模板时必须小心,因为它在标准的 panic 恢复陷阱之外运行。此模板应该尽可能简单,因为它会为每个请求运行一次。
|
|
||||||
|
|
||||||
## 释放和重新打开、暂停和恢复日志记录
|
|
||||||
|
|
||||||
如果您在 Unix 上运行,您可能希望释放和重新打开日志以使用 `logrotate` 或其他工具。
|
|
||||||
可以通过向运行中的进程发送 `SIGUSR1` 信号或运行 `gitea manager logging release-and-reopen` 命令来强制 Gitea 释放并重新打开其日志文件和连接。
|
|
||||||
|
|
||||||
或者,您可能希望暂停和恢复日志记录 - 可以通过使用 `gitea manager logging pause` 和 `gitea manager logging resume` 命令来实现。请注意,当日志记录暂停时,低于 INFO 级别的日志事件将不会存储,并且只会存储有限数量的事件。在暂停时,日志记录可能会阻塞,尽管是暂时性的,但会大大减慢 Gitea 的运行速度,因此建议仅暂停很短的时间。
|
|
||||||
|
|
||||||
### 在 Gitea 运行时添加和删除日志记录
|
|
||||||
|
|
||||||
可以使用 `gitea manager logging add` 和 `remove` 子命令在 Gitea 运行时添加和删除日志记录。
|
|
||||||
此功能只能调整正在运行的日志系统,不能用于启动未初始化的访问或路由日志记录器。如果您希望启动这些系统,建议调整 app.ini 并(优雅地)重新启动 Gitea 服务。
|
|
||||||
|
|
||||||
这些命令的主要目的是在运行中的系统上轻松添加临时日志记录器,以便调查问题,因为重新启动可能会导致问题消失。
|
|
||||||
|
|
||||||
## 使用 `logrotate` 而不是内置的日志轮转
|
|
||||||
|
|
||||||
Gitea 包含内置的日志轮转功能,对于大多数部署来说应该已经足够了。但是,如果您想使用 `logrotate` 工具:
|
|
||||||
|
|
||||||
- 在 `app.ini` 中将 `LOG_ROTATE` 设置为 `false`,禁用内置的日志轮转。
|
|
||||||
- 安装 `logrotate`。
|
|
||||||
- 根据部署要求配置 `logrotate`,有关配置语法细节,请参阅 `man 8 logrotate`。
|
|
||||||
在 `postrotate/endscript` 块中通过 `kill -USR1` 或 `kill -10` 向 `gitea` 进程本身发送 `USR1` 信号,
|
|
||||||
或者运行 `gitea manager logging release-and-reopen`(使用适当的环境设置)。
|
|
||||||
确保配置适用于由 Gitea 日志记录器生成的所有文件,如上述部分所述。
|
|
||||||
- 始终使用 `logrotate /etc/logrotate.conf --debug` 来测试您的配置。
|
|
||||||
- 如果您正在使用 Docker 并从容器外部运行,您可以使用
|
|
||||||
`docker exec -u $OS_USER $CONTAINER_NAME sh -c 'gitea manager logging release-and-reopen'`
|
|
||||||
或 `docker exec $CONTAINER_NAME sh -c '/bin/s6-svc -1 /etc/s6/gitea/'`,或直接向 Gitea 进程本身发送 `USR1` 信号。
|
|
||||||
|
|
||||||
下一个 `logrotate` 作业将包括您的配置,因此不需要重新启动。
|
|
||||||
您还可以立即使用 `logrotate /etc/logrotate.conf --force` 重新加载 `logrotate`。
|
|
@ -1,278 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2019-10-23T17:00:00-03:00"
|
|
||||||
title: "Mail templates"
|
|
||||||
slug: "mail-templates"
|
|
||||||
sidebar_position: 45
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/mail-templates
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Mail templates"
|
|
||||||
sidebar_position: 45
|
|
||||||
identifier: "mail-templates"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Mail templates
|
|
||||||
|
|
||||||
To craft the e-mail subject and contents for certain operations, Gitea can be customized by using templates. The templates
|
|
||||||
for these functions are located under the [`custom` directory](administration/customizing-gitea.md).
|
|
||||||
Gitea has an internal template that serves as default in case there's no custom alternative.
|
|
||||||
|
|
||||||
Custom templates are loaded when Gitea starts. Changes made to them are not recognized until Gitea is restarted again.
|
|
||||||
|
|
||||||
## Mail notifications supporting templates
|
|
||||||
|
|
||||||
Currently, the following notification events make use of templates:
|
|
||||||
|
|
||||||
| Action name | Usage |
|
|
||||||
| ----------- | ------------------------------------------------------------------------------------------------------------ |
|
|
||||||
| `new` | A new issue or pull request was created. |
|
|
||||||
| `comment` | A new comment was created in an existing issue or pull request. |
|
|
||||||
| `close` | An issue or pull request was closed. |
|
|
||||||
| `reopen` | An issue or pull request was reopened. |
|
|
||||||
| `review` | The head comment of a review in a pull request. |
|
|
||||||
| `approve` | The head comment of a approving review for a pull request. |
|
|
||||||
| `reject` | The head comment of a review requesting changes for a pull request. |
|
|
||||||
| `code` | A single comment on the code of a pull request. |
|
|
||||||
| `assigned` | User was assigned to an issue or pull request. |
|
|
||||||
| `default` | Any action not included in the above categories, or when the corresponding category template is not present. |
|
|
||||||
|
|
||||||
The path for the template of a particular message type is:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
custom/templates/mail/{action type}/{action name}.tmpl
|
|
||||||
```
|
|
||||||
|
|
||||||
Where `{action type}` is one of `issue` or `pull` (for pull requests), and `{action name}` is one of the names listed above.
|
|
||||||
|
|
||||||
For example, the specific template for a mail regarding a comment in a pull request is:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
custom/templates/mail/pull/comment.tmpl
|
|
||||||
```
|
|
||||||
|
|
||||||
However, creating templates for each and every action type/name combination is not required.
|
|
||||||
A fallback system is used to choose the appropriate template for an event. The _first existing_
|
|
||||||
template on this list is used:
|
|
||||||
|
|
||||||
- The specific template for the desired **action type** and **action name**.
|
|
||||||
- The template for action type `issue` and the desired **action name**.
|
|
||||||
- The template for the desired **action type**, action name `default`.
|
|
||||||
- The template for action type `issue`, action name `default`.
|
|
||||||
|
|
||||||
The only mandatory template is action type `issue`, action name `default`, which is already embedded in Gitea
|
|
||||||
unless it's overridden by the user in the `custom` directory.
|
|
||||||
|
|
||||||
## Template syntax
|
|
||||||
|
|
||||||
Mail templates are UTF-8 encoded text files that need to follow one of the following formats:
|
|
||||||
|
|
||||||
```
|
|
||||||
Text and macros for the subject line
|
|
||||||
------------
|
|
||||||
Text and macros for the mail body
|
|
||||||
```
|
|
||||||
|
|
||||||
or
|
|
||||||
|
|
||||||
```
|
|
||||||
Text and macros for the mail body
|
|
||||||
```
|
|
||||||
|
|
||||||
Specifying a _subject_ section is optional (and therefore also the dash line separator). When used, the separator between
|
|
||||||
_subject_ and _mail body_ templates requires at least three dashes; no other characters are allowed in the separator line.
|
|
||||||
|
|
||||||
_Subject_ and _mail body_ are parsed by [Golang's template engine](https://go.dev/pkg/text/template/) and
|
|
||||||
are provided with a _metadata context_ assembled for each notification. The context contains the following elements:
|
|
||||||
|
|
||||||
| Name | Type | Available | Usage |
|
|
||||||
| ------------------ | ---------------- | ------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
|
|
||||||
| `.FallbackSubject` | string | Always | A default subject line. See Below. |
|
|
||||||
| `.Subject` | string | Only in body | The _subject_, once resolved. |
|
|
||||||
| `.Body` | string | Always | The message of the issue, pull request or comment, parsed from Markdown into HTML and sanitized. Do not confuse with the _mail body_. |
|
|
||||||
| `.Link` | string | Always | The address of the originating issue, pull request or comment. |
|
|
||||||
| `.Issue` | models.Issue | Always | The issue (or pull request) originating the notification. To get data specific to a pull request (e.g. `HasMerged`), `.Issue.PullRequest` can be used, but care should be taken as this field will be `nil` if the issue is _not_ a pull request. |
|
|
||||||
| `.Comment` | models.Comment | If applicable | If the notification is from a comment added to an issue or pull request, this will contain the information about the comment. |
|
|
||||||
| `.IsPull` | bool | Always | `true` if the mail notification is associated with a pull request (i.e. `.Issue.PullRequest` is not `nil`). |
|
|
||||||
| `.Repo` | string | Always | Name of the repository, including owner name (e.g. `mike/stuff`) |
|
|
||||||
| `.User` | models.User | Always | Owner of the repository from which the event originated. To get the user name (e.g. `mike`),`.User.Name` can be used. |
|
|
||||||
| `.Doer` | models.User | Always | User that executed the action triggering the notification event. To get the user name (e.g. `rhonda`), `.Doer.Name` can be used. |
|
|
||||||
| `.IsMention` | bool | Always | `true` if this notification was only generated because the user was mentioned in the comment, while not being subscribed to the source. It will be `false` if the recipient was subscribed to the issue or repository. |
|
|
||||||
| `.SubjectPrefix` | string | Always | `Re: ` if the notification is about other than issue or pull request creation; otherwise an empty string. |
|
|
||||||
| `.ActionType` | string | Always | `"issue"` or `"pull"`. Will correspond to the actual _action type_ independently of which template was selected. |
|
|
||||||
| `.ActionName` | string | Always | It will be one of the action types described above (`new`, `comment`, etc.), and will correspond to the actual _action name_ independently of which template was selected. |
|
|
||||||
| `.ReviewComments` | []models.Comment | Always | List of code comments in a review. The comment text will be in `.RenderedContent` and the referenced code will be in `.Patch`. |
|
|
||||||
|
|
||||||
All names are case sensitive.
|
|
||||||
|
|
||||||
### The _subject_ part of the template
|
|
||||||
|
|
||||||
The template engine used for the mail _subject_ is golang's [`text/template`](https://go.dev/pkg/text/template/).
|
|
||||||
Please refer to the linked documentation for details about its syntax.
|
|
||||||
|
|
||||||
The _subject_ is built using the following steps:
|
|
||||||
|
|
||||||
- A template is selected according to the type of notification and to what templates are present.
|
|
||||||
- The template is parsed and resolved (e.g. `{{.Issue.Index}}` is converted to the number of the issue
|
|
||||||
or pull request).
|
|
||||||
- All space-like characters (e.g. `TAB`, `LF`, etc.) are converted to normal spaces.
|
|
||||||
- All leading, trailing and redundant spaces are removed.
|
|
||||||
- The string is truncated to its first 256 runes (characters).
|
|
||||||
|
|
||||||
If the end result is an empty string, **or** no subject template was available (i.e. the selected template
|
|
||||||
did not include a subject part), Gitea's **internal default** will be used.
|
|
||||||
|
|
||||||
The internal default (fallback) subject is the equivalent of:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
{{.SubjectPrefix}}[{{.Repo}}] {{.Issue.Title}} (#{{.Issue.Index}})
|
|
||||||
```
|
|
||||||
|
|
||||||
For example: `Re: [mike/stuff] New color palette (#38)`
|
|
||||||
|
|
||||||
Gitea's default subject can also be found in the template _metadata_ as `.FallbackSubject` from any of
|
|
||||||
the two templates, even if a valid subject template is present.
|
|
||||||
|
|
||||||
### The _mail body_ part of the template
|
|
||||||
|
|
||||||
The template engine used for the _mail body_ is golang's [`html/template`](https://go.dev/pkg/html/template/).
|
|
||||||
Please refer to the linked documentation for details about its syntax.
|
|
||||||
|
|
||||||
The _mail body_ is parsed after the mail subject, so there is an additional _metadata_ field which is
|
|
||||||
the actual rendered subject, after all considerations.
|
|
||||||
|
|
||||||
The expected result is HTML (including structural elements like`<html>`, `<body>`, etc.). Styling
|
|
||||||
through `<style>` blocks, `class` and `style` attributes is possible. However, `html/template`
|
|
||||||
does some [automatic escaping](https://go.dev/pkg/html/template/#hdr-Contexts) that should be considered.
|
|
||||||
|
|
||||||
Attachments (such as images or external style sheets) are not supported. However, other templates can
|
|
||||||
be referenced too, for example to provide the contents of a `<style>` element in a centralized fashion.
|
|
||||||
The external template must be placed under `custom/mail` and referenced relative to that directory.
|
|
||||||
For example, `custom/mail/styles/base.tmpl` can be included using `{{template styles/base}}`.
|
|
||||||
|
|
||||||
The mail is sent with `Content-Type: multipart/alternative`, so the body is sent in both HTML
|
|
||||||
and text formats. The latter is obtained by stripping the HTML markup.
|
|
||||||
|
|
||||||
## Troubleshooting
|
|
||||||
|
|
||||||
How a mail is rendered is directly dependent on the capabilities of the mail application. Many mail
|
|
||||||
clients don't even support HTML, so they show the text version included in the generated mail.
|
|
||||||
|
|
||||||
If the template fails to render, it will be noticed only at the moment the mail is sent.
|
|
||||||
A default subject is used if the subject template fails, and whatever was rendered successfully
|
|
||||||
from the _mail body_ is used, disregarding the rest.
|
|
||||||
|
|
||||||
Please check [Gitea's logs](administration/logging-config.md) for error messages in case of trouble.
|
|
||||||
|
|
||||||
## Example
|
|
||||||
|
|
||||||
`custom/templates/mail/issue/default.tmpl`:
|
|
||||||
|
|
||||||
```html
|
|
||||||
[{{.Repo}}] @{{.Doer.Name}}
|
|
||||||
{{if eq .ActionName "new"}}
|
|
||||||
created
|
|
||||||
{{else if eq .ActionName "comment"}}
|
|
||||||
commented on
|
|
||||||
{{else if eq .ActionName "close"}}
|
|
||||||
closed
|
|
||||||
{{else if eq .ActionName "reopen"}}
|
|
||||||
reopened
|
|
||||||
{{else}}
|
|
||||||
updated
|
|
||||||
{{end}}
|
|
||||||
{{if eq .ActionType "issue"}}
|
|
||||||
issue
|
|
||||||
{{else}}
|
|
||||||
pull request
|
|
||||||
{{end}}
|
|
||||||
#{{.Issue.Index}}: {{.Issue.Title}}
|
|
||||||
------------
|
|
||||||
<!DOCTYPE html>
|
|
||||||
<html>
|
|
||||||
<head>
|
|
||||||
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
|
|
||||||
<title>{{.Subject}}</title>
|
|
||||||
</head>
|
|
||||||
|
|
||||||
<body>
|
|
||||||
{{if .IsMention}}
|
|
||||||
<p>
|
|
||||||
You are receiving this because @{{.Doer.Name}} mentioned you.
|
|
||||||
</p>
|
|
||||||
{{end}}
|
|
||||||
<p>
|
|
||||||
<p>
|
|
||||||
<a href="{{AppUrl}}/{{.Doer.LowerName}}">@{{.Doer.Name}}</a>
|
|
||||||
{{if not (eq .Doer.FullName "")}}
|
|
||||||
({{.Doer.FullName}})
|
|
||||||
{{end}}
|
|
||||||
{{if eq .ActionName "new"}}
|
|
||||||
created
|
|
||||||
{{else if eq .ActionName "close"}}
|
|
||||||
closed
|
|
||||||
{{else if eq .ActionName "reopen"}}
|
|
||||||
reopened
|
|
||||||
{{else}}
|
|
||||||
updated
|
|
||||||
{{end}}
|
|
||||||
<a href="{{.Link}}">{{.Repo}}#{{.Issue.Index}}</a>.
|
|
||||||
</p>
|
|
||||||
{{if not (eq .Body "")}}
|
|
||||||
<h3>Message content</h3>
|
|
||||||
<hr>
|
|
||||||
{{.Body}}
|
|
||||||
{{end}}
|
|
||||||
</p>
|
|
||||||
<hr>
|
|
||||||
<p>
|
|
||||||
<a href="{{.Link}}">View it on Gitea</a>.
|
|
||||||
</p>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
```
|
|
||||||
|
|
||||||
This template produces something along these lines:
|
|
||||||
|
|
||||||
### Subject
|
|
||||||
|
|
||||||
> [mike/stuff] @rhonda commented on pull request #38: New color palette
|
|
||||||
|
|
||||||
### Mail body
|
|
||||||
|
|
||||||
> [@rhonda](#) (Rhonda Myers) updated [mike/stuff#38](#).
|
|
||||||
>
|
|
||||||
> #### Message content
|
|
||||||
>
|
|
||||||
> \_********************************\_********************************
|
|
||||||
>
|
|
||||||
> Mike, I think we should tone down the blues a little.
|
|
||||||
> \_********************************\_********************************
|
|
||||||
>
|
|
||||||
> [View it on Gitea](#).
|
|
||||||
|
|
||||||
## Advanced
|
|
||||||
|
|
||||||
The template system contains several functions that can be used to further process and format
|
|
||||||
the messages. Here's a list of some of them:
|
|
||||||
|
|
||||||
| Name | Parameters | Available | Usage |
|
|
||||||
| ---------------- | ----------- | --------- | ------------------------------------------------------------------- |
|
|
||||||
| `AppUrl` | - | Any | Gitea's URL |
|
|
||||||
| `AppName` | - | Any | Set from `app.ini`, usually "Gitea" |
|
|
||||||
| `AppDomain` | - | Any | Gitea's host name |
|
|
||||||
| `EllipsisString` | string, int | Any | Truncates a string to the specified length; adds ellipsis as needed |
|
|
||||||
| `SanitizeHTML` | string | Body only | Sanitizes text by removing any dangerous HTML tags from it |
|
|
||||||
| `SafeHTML` | string | Body only | Takes the input as HTML, can be used for outputing raw HTML content |
|
|
||||||
|
|
||||||
These are _functions_, not metadata, so they have to be used:
|
|
||||||
|
|
||||||
```html
|
|
||||||
Like this: {{SanitizeHTML "Escape<my>text"}}
|
|
||||||
Or this: {{"Escape<my>text" | SanitizeHTML}}
|
|
||||||
Or this: {{AppUrl}}
|
|
||||||
But not like this: {{.AppUrl}}
|
|
||||||
```
|
|
@ -1,261 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2023-05-23T09:00:00+08:00"
|
|
||||||
title: "邮件模板"
|
|
||||||
slug: "mail-templates"
|
|
||||||
sidebar_position: 45
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/mail-templates
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "邮件模板"
|
|
||||||
sidebar_position: 45
|
|
||||||
identifier: "mail-templates"
|
|
||||||
---
|
|
||||||
|
|
||||||
# 邮件模板
|
|
||||||
|
|
||||||
为了定制特定操作的电子邮件主题和内容,可以使用模板来自定义 Gitea。这些功能的模板位于 [`custom` 目录](administration/customizing-gitea.md) 下。
|
|
||||||
如果没有自定义的替代方案,Gitea 将使用内部模板作为默认模板。
|
|
||||||
|
|
||||||
自定义模板在 Gitea 启动时加载。对它们的更改在 Gitea 重新启动之前不会被识别。
|
|
||||||
|
|
||||||
## 支持模板的邮件通知
|
|
||||||
|
|
||||||
目前,以下通知事件使用模板:
|
|
||||||
|
|
||||||
| 操作名称 | 用途 |
|
|
||||||
| ----------- | ------------------------------------------------------------------------------------------------------------ |
|
|
||||||
| `new` | 创建了新的工单或合并请求。 |
|
|
||||||
| `comment` | 在现有工单或合并请求中创建了新的评论。 |
|
|
||||||
| `close` | 关闭了工单或合并请求。 |
|
|
||||||
| `reopen` | 重新打开了工单或合并请求。 |
|
|
||||||
| `review` | 在合并请求中进行审查的首要评论。 |
|
|
||||||
| `approve` | 对合并请求进行批准的首要评论。 |
|
|
||||||
| `reject` | 对合并请求提出更改请求的审查的首要评论。 |
|
|
||||||
| `code` | 关于合并请求的代码的单个评论。 |
|
|
||||||
| `assigned` | 用户被分配到工单或合并请求。 |
|
|
||||||
| `default` | 未包括在上述类别中的任何操作,或者当对应类别的模板不存在时使用的模板。 |
|
|
||||||
|
|
||||||
特定消息类型的模板路径为:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
custom/templates/mail/{操作类型}/{操作名称}.tmpl
|
|
||||||
```
|
|
||||||
|
|
||||||
其中 `{操作类型}` 是 `issue` 或 `pull`(针对合并请求),`{操作名称}` 是上述列出的操作名称之一。
|
|
||||||
|
|
||||||
例如,有关合并请求中的评论的电子邮件的特定模板是:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
custom/templates/mail/pull/comment.tmpl
|
|
||||||
```
|
|
||||||
|
|
||||||
然而,并不需要为每个操作类型/名称组合创建模板。
|
|
||||||
使用回退系统来选择适当的模板。在此列表中,将使用 _第一个存在的_ 模板:
|
|
||||||
|
|
||||||
- 所需**操作类型**和**操作名称**的特定模板。
|
|
||||||
- 操作类型为 `issue` 和所需**操作名称**的模板。
|
|
||||||
- 所需**操作类型**和操作名称为 `default` 的模板。
|
|
||||||
- 操作类型为` issue` 和操作名称为 `default` 的模板。
|
|
||||||
|
|
||||||
唯一必需的模板是操作类型为 `issue` 操作名称为 `default` 的模板,除非用户在 `custom` 目录中覆盖了它。
|
|
||||||
|
|
||||||
## 模板语法
|
|
||||||
|
|
||||||
邮件模板是 UTF-8 编码的文本文件,需要遵循以下格式之一:
|
|
||||||
|
|
||||||
```
|
|
||||||
用于主题行的文本和宏
|
|
||||||
------------
|
|
||||||
用于邮件正文的文本和宏
|
|
||||||
```
|
|
||||||
|
|
||||||
或者
|
|
||||||
|
|
||||||
```
|
|
||||||
用于邮件正文的文本和宏
|
|
||||||
```
|
|
||||||
|
|
||||||
指定 _主题_ 部分是可选的(因此也是虚线分隔符)。在使用时,_主题_ 和 _邮件正文_ 模板之间的分隔符需要至少三个虚线;分隔符行中不允许使用其他字符。
|
|
||||||
|
|
||||||
_主题_ 和 _邮件正文_ 由 [Golang的模板引擎](https://go.dev/pkg/text/template/) 解析,并提供了为每个通知组装的 _元数据上下文_。上下文包含以下元素:
|
|
||||||
|
|
||||||
| 名称 | 类型 | 可用性 | 用途 |
|
|
||||||
| -------------------- | ------------------ | ----------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
|
|
||||||
| `.FallbackSubject` | string | 始终可用 | 默认主题行。参见下文。 |
|
|
||||||
| `.Subject` | string | 仅在正文中可用 | 解析后的 _主题_。 |
|
|
||||||
| `.Body` | string | 始终可用 | 工单、合并请求或评论的消息,从 Markdown 解析为 HTML 并进行了清理。请勿与 _邮件正文_ 混淆。 |
|
|
||||||
| `.Link` | string | 始终可用 | 源工单、合并请求或评论的地址。 |
|
|
||||||
| `.Issue` | models.Issue | 始终可用 | 产生通知的工单(或合并请求)。要获取特定于合并请求的数据(例如 `HasMerged`),可以使用 `.Issue.PullRequest`,但需要注意,如果工单 _不是_ 合并请求,则该字段将为 `nil`。 |
|
|
||||||
| `.Comment` | models.Comment | 如果适用 | 如果通知是针对添加到工单或合并请求的评论,则其中包含有关评论的信息。 |
|
|
||||||
| `.IsPull` | bool | 始终可用 | 如果邮件通知与合并请求关联(即 `.Issue.PullRequest` 不为 `nil` ),则为 `true`。 |
|
|
||||||
| `.Repo` | string | 始终可用 | 仓库的名称,包括所有者名称(例如 `mike/stuff`) |
|
|
||||||
| `.User` | models.User | 始终可用 | 事件来源仓库的所有者。要获取用户名(例如 `mike`),可以使用 `.User.Name`。 |
|
|
||||||
| `.Doer` | models.User | 始终可用 | 执行触发通知事件的操作的用户。要获取用户名(例如 `rhonda`),可以使用 `.Doer.Name`。 |
|
|
||||||
| `.IsMention` | bool | 始终可用 | 如果此通知仅是因为在评论中提到了用户而生成的,并且收件人未订阅源,则为 `true`。如果收件人已订阅工单或仓库,则为 `false`。 |
|
|
||||||
| `.SubjectPrefix` | string | 始终可用 | 如果通知是关于除工单或合并请求创建之外的其他内容,则为 `Re:`;否则为空字符串。 |
|
|
||||||
| `.ActionType` | string | 始终可用 | `"issue"` 或 `"pull"`。它将与实际的 _操作类型_ 对应,与选择的模板无关。 |
|
|
||||||
| `.ActionName` | string | 始终可用 | 它将是上述操作类型之一(`new` ,`comment` 等),并与选择的模板对应。 |
|
|
||||||
| `.ReviewComments` | []models.Comment | 始终可用 | 审查中的代码评论列表。评论文本将在 `.RenderedContent` 中,引用的代码将在 `.Patch` 中。 |
|
|
||||||
|
|
||||||
所有名称区分大小写。
|
|
||||||
|
|
||||||
### 模板中的主题部分
|
|
||||||
|
|
||||||
用于邮件主题的模板引擎是 Golang 的 [`text/template`](https://go.dev/pkg/text/template/)。
|
|
||||||
有关语法的详细信息,请参阅链接的文档。
|
|
||||||
|
|
||||||
主题构建的步骤如下:
|
|
||||||
|
|
||||||
- 根据通知类型和可用的模板选择一个模板。
|
|
||||||
- 解析并解析模板(例如,将 `{{.Issue.Index}}` 转换为工单或合并请求的编号)。
|
|
||||||
- 将所有空格字符(例如 `TAB`,`LF` 等)转换为普通空格。
|
|
||||||
- 删除所有前导、尾随和多余的空格。
|
|
||||||
- 将字符串截断为前 256 个字母(字符)。
|
|
||||||
|
|
||||||
如果最终结果为空字符串,**或者**没有可用的主题模板(即所选模板不包含主题部分),将使用Gitea的**内部默认值**。
|
|
||||||
|
|
||||||
内部默认(回退)主题相当于:
|
|
||||||
|
|
||||||
```
|
|
||||||
{{.SubjectPrefix}}[{{.Repo}}] {{.Issue.Title}} (#{{.Issue.Index}})
|
|
||||||
```
|
|
||||||
|
|
||||||
例如:`Re: [mike/stuff] New color palette (#38)`
|
|
||||||
|
|
||||||
即使存在有效的主题模板,Gitea的默认主题也可以在模板的元数据中作为 `.FallbackSubject` 找到。
|
|
||||||
|
|
||||||
### 模板中的邮件正文部分
|
|
||||||
|
|
||||||
用于邮件正文的模板引擎是 Golang 的 [`html/template`](https://go.dev/pkg/html/template/)。
|
|
||||||
有关语法的详细信息,请参阅链接的文档。
|
|
||||||
|
|
||||||
邮件正文在邮件主题之后进行解析,因此还有一个额外的 _元数据_ 字段,即在考虑所有情况之后实际呈现的主题。
|
|
||||||
|
|
||||||
期望的结果是 HTML(包括结构元素,如`<html>`,`<body>`等)。可以通过 `<style>` 块、`class` 和 `style` 属性进行样式设置。但是,`html/template` 会进行一些 [自动转义](https://go.dev/pkg/html/template/#hdr-Contexts),需要考虑这一点。
|
|
||||||
|
|
||||||
不支持附件(例如图像或外部样式表)。但是,也可以引用其他模板,例如以集中方式提供 `<style>` 元素的内容。外部模板必须放置在 `custom/mail` 下,并相对于该目录引用。例如,可以使用 `{{template styles/base}}` 包含 `custom/mail/styles/base.tmpl`。
|
|
||||||
|
|
||||||
邮件以 `Content-Type: multipart/alternative` 发送,因此正文以 HTML 和文本格式发送。通过剥离 HTML 标记来获取文本版本。
|
|
||||||
|
|
||||||
## 故障排除
|
|
||||||
|
|
||||||
邮件的呈现方式直接取决于邮件应用程序的功能。许多邮件客户端甚至不支持 HTML,因此显示生成邮件中包含的文本版本。
|
|
||||||
|
|
||||||
如果模板无法呈现,则只有在发送邮件时才会注意到。
|
|
||||||
如果主题模板失败,将使用默认主题,如果从 _邮件正文_ 中成功呈现了任何内容,则将使用该内容,忽略其他内容。
|
|
||||||
|
|
||||||
如果遇到问题,请检查 [Gitea的日志](administration/logging-config.md) 以获取错误消息。
|
|
||||||
|
|
||||||
## 示例
|
|
||||||
|
|
||||||
`custom/templates/mail/issue/default.tmpl`:
|
|
||||||
|
|
||||||
```html
|
|
||||||
[{{.Repo}}] @{{.Doer.Name}}
|
|
||||||
{{if eq .ActionName "new"}}
|
|
||||||
创建了
|
|
||||||
{{else if eq .ActionName "comment"}}
|
|
||||||
评论了
|
|
||||||
{{else if eq .ActionName "close"}}
|
|
||||||
关闭了
|
|
||||||
{{else if eq .ActionName "reopen"}}
|
|
||||||
重新打开了
|
|
||||||
{{else}}
|
|
||||||
更新了
|
|
||||||
{{end}}
|
|
||||||
{{if eq .ActionType "issue"}}
|
|
||||||
工单
|
|
||||||
{{else}}
|
|
||||||
合并请求
|
|
||||||
{{end}}
|
|
||||||
#{{.Issue.Index}}: {{.Issue.Title}}
|
|
||||||
------------
|
|
||||||
<!DOCTYPE html>
|
|
||||||
<html>
|
|
||||||
<head>
|
|
||||||
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
|
|
||||||
<title>{{.Subject}}</title>
|
|
||||||
</head>
|
|
||||||
|
|
||||||
<body>
|
|
||||||
{{if .IsMention}}
|
|
||||||
<p>
|
|
||||||
您收到此邮件是因为 @{{.Doer.Name}} 提到了您。
|
|
||||||
</p>
|
|
||||||
{{end}}
|
|
||||||
<p>
|
|
||||||
<p>
|
|
||||||
<a href="{{AppUrl}}/{{.Doer.LowerName}}">@{{.Doer.Name}}</a>
|
|
||||||
{{if not (eq .Doer.FullName "")}}
|
|
||||||
({{.Doer.FullName}})
|
|
||||||
{{end}}
|
|
||||||
{{if eq .ActionName "new"}}
|
|
||||||
创建了
|
|
||||||
{{else if eq .ActionName "close"}}
|
|
||||||
关闭了
|
|
||||||
{{else if eq .ActionName "reopen"}}
|
|
||||||
重新打开了
|
|
||||||
{{else}}
|
|
||||||
更新了
|
|
||||||
{{end}}
|
|
||||||
<a href="{{.Link}}">{{.Repo}}#{{.Issue.Index}}</a>。
|
|
||||||
</p>
|
|
||||||
{{if not (eq .Body "")}}
|
|
||||||
<h3>消息内容:</h3>
|
|
||||||
<hr>
|
|
||||||
{{.Body}}
|
|
||||||
{{end}}
|
|
||||||
</p>
|
|
||||||
<hr>
|
|
||||||
<p>
|
|
||||||
<a href="{{.Link}}">在 Gitea 上查看</a>。
|
|
||||||
</p>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
```
|
|
||||||
|
|
||||||
该模板将生成以下内容:
|
|
||||||
|
|
||||||
### 主题
|
|
||||||
|
|
||||||
> [mike/stuff] @rhonda 在合并请求 #38 上进行了评论:New color palette
|
|
||||||
|
|
||||||
### 邮件正文
|
|
||||||
|
|
||||||
> [@rhonda](#)(Rhonda Myers)更新了 [mike/stuff#38](#)。
|
|
||||||
>
|
|
||||||
> #### 消息内容
|
|
||||||
>
|
|
||||||
> \_********************************\_********************************
|
|
||||||
>
|
|
||||||
> Mike, I think we should tone down the blues a little.
|
|
||||||
>
|
|
||||||
> \_********************************\_********************************
|
|
||||||
>
|
|
||||||
> [在 Gitea 上查看](#)。
|
|
||||||
|
|
||||||
## 高级用法
|
|
||||||
|
|
||||||
模板系统包含一些函数,可用于进一步处理和格式化消息。以下是其中一些函数的列表:
|
|
||||||
|
|
||||||
| 函数名 | 参数 | 可用于 | 用法 |
|
|
||||||
|------------------| ----------- | ------------ | ------------------------------ |
|
|
||||||
| `AppUrl` | - | 任何地方 | Gitea 的 URL |
|
|
||||||
| `AppName` | - | 任何地方 | 从 `app.ini` 中设置,通常为 "Gitea" |
|
|
||||||
| `AppDomain` | - | 任何地方 | Gitea 的主机名 |
|
|
||||||
| `EllipsisString` | string, int | 任何地方 | 将字符串截断为指定长度;根据需要添加省略号 |
|
|
||||||
| `SanitizeHTML` | string | 仅正文部分 | 通过删除其中的危险 HTML 标签对文本进行清理 |
|
|
||||||
| `SafeHTML` | string | 仅正文部分 | 将输入作为 HTML 处理;可用于输出原始的 HTML 内容 |
|
|
||||||
|
|
||||||
这些都是 _函数_,而不是元数据,因此必须按以下方式使用:
|
|
||||||
|
|
||||||
```html
|
|
||||||
像这样使用: {{SanitizeHTML "Escape<my>text"}}
|
|
||||||
或者这样使用: {{"Escape<my>text" | SanitizeHTML}}
|
|
||||||
或者这样使用: {{AppUrl}}
|
|
||||||
但不要像这样使用: {{.AppUrl}}
|
|
||||||
```
|
|
@ -1,65 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2019-09-06T01:35:00-03:00"
|
|
||||||
title: "Repository indexer"
|
|
||||||
slug: "repo-indexer"
|
|
||||||
sidebar_position: 45
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/repo-indexer
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Repository indexer"
|
|
||||||
sidebar_position: 45
|
|
||||||
identifier: "repo-indexer"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Repository indexer
|
|
||||||
|
|
||||||
## Builtin repository code search without indexer
|
|
||||||
|
|
||||||
Users could do repository-level code search without setting up a repository indexer.
|
|
||||||
The builtin code search is based on the `git grep` command, which is fast and efficient for small repositories.
|
|
||||||
Better code search support could be achieved by setting up the repository indexer.
|
|
||||||
|
|
||||||
## Setting up the repository indexer
|
|
||||||
|
|
||||||
Gitea can search through the files of the repositories by enabling this function in your [`app.ini`](administration/config-cheat-sheet.md):
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[indexer]
|
|
||||||
; ...
|
|
||||||
REPO_INDEXER_ENABLED = true
|
|
||||||
REPO_INDEXER_PATH = indexers/repos.bleve
|
|
||||||
MAX_FILE_SIZE = 1048576
|
|
||||||
REPO_INDEXER_INCLUDE =
|
|
||||||
REPO_INDEXER_EXCLUDE = resources/bin/**
|
|
||||||
```
|
|
||||||
|
|
||||||
Please bear in mind that indexing the contents can consume a lot of system resources, especially when the index is created for the first time or globally updated (e.g. after upgrading Gitea).
|
|
||||||
|
|
||||||
### Choosing the files for indexing by size
|
|
||||||
|
|
||||||
The `MAX_FILE_SIZE` option will make the indexer skip all files larger than the specified value.
|
|
||||||
|
|
||||||
### Choosing the files for indexing by path
|
|
||||||
|
|
||||||
Gitea applies glob pattern matching from the [`gobwas/glob` library](https://github.com/gobwas/glob) to choose which files will be included in the index.
|
|
||||||
|
|
||||||
Limiting the list of files prevents the indexes from becoming polluted with derived or irrelevant files (e.g. lss, sym, map, etc.), so the search results are more relevant. It can also help reduce the index size.
|
|
||||||
|
|
||||||
`REPO_INDEXER_EXCLUDE_VENDORED` (default: true) excludes vendored files from index.
|
|
||||||
|
|
||||||
`REPO_INDEXER_INCLUDE` (default: empty) is a comma separated list of glob patterns to **include** in the index. An empty list means "_include all files_".
|
|
||||||
`REPO_INDEXER_EXCLUDE` (default: empty) is a comma separated list of glob patterns to **exclude** from the index. Files that match this list will not be indexed. `REPO_INDEXER_EXCLUDE` takes precedence over `REPO_INDEXER_INCLUDE`.
|
|
||||||
|
|
||||||
Pattern matching works as follows:
|
|
||||||
|
|
||||||
- To match all files with a `.txt` extension no matter what directory, use `**.txt`.
|
|
||||||
- To match all files with a `.txt` extension _only at the root level of the repository_, use `*.txt`.
|
|
||||||
- To match all files inside `resources/bin` and below, use `resources/bin/**`.
|
|
||||||
- To match all files _immediately inside_ `resources/bin`, use `resources/bin/*`.
|
|
||||||
- To match all files named `Makefile`, use `**Makefile`.
|
|
||||||
- Matching a directory has no effect; the pattern `resources/bin` will not include/exclude files inside that directory; `resources/bin/**` will.
|
|
||||||
- All files and patterns are normalized to lower case, so `**Makefile`, `**makefile` and `**MAKEFILE` are equivalent.
|
|
@ -1,59 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2023-05-23T09:00:00+08:00"
|
|
||||||
title: "仓库索引器"
|
|
||||||
slug: "repo-indexer"
|
|
||||||
sidebar_position: 45
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /zh-cn/repo-indexer
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "仓库索引器"
|
|
||||||
sidebar_position: 45
|
|
||||||
identifier: "repo-indexer"
|
|
||||||
---
|
|
||||||
|
|
||||||
# 仓库索引器
|
|
||||||
|
|
||||||
## 设置仓库索引器
|
|
||||||
|
|
||||||
通过在您的 [`app.ini`](administration/config-cheat-sheet.md) 中启用此功能,Gitea 可以通过仓库的文件进行搜索:
|
|
||||||
|
|
||||||
```ini
|
|
||||||
[indexer]
|
|
||||||
; ...
|
|
||||||
REPO_INDEXER_ENABLED = true
|
|
||||||
REPO_INDEXER_PATH = indexers/repos.bleve
|
|
||||||
MAX_FILE_SIZE = 1048576
|
|
||||||
REPO_INDEXER_INCLUDE =
|
|
||||||
REPO_INDEXER_EXCLUDE = resources/bin/**
|
|
||||||
```
|
|
||||||
|
|
||||||
请记住,索引内容可能会消耗大量系统资源,特别是在首次创建索引或全局更新索引时(例如升级 Gitea 之后)。
|
|
||||||
|
|
||||||
### 按大小选择要索引的文件
|
|
||||||
|
|
||||||
`MAX_FILE_SIZE` 选项将使索引器跳过所有大于指定值的文件。
|
|
||||||
|
|
||||||
### 按路径选择要索引的文件
|
|
||||||
|
|
||||||
Gitea使用 [`gobwas/glob` 库](https://github.com/gobwas/glob) 中的 glob 模式匹配来选择要包含在索引中的文件。
|
|
||||||
|
|
||||||
限制文件列表可以防止索引被派生或无关的文件(例如 lss、sym、map 等)污染,从而使搜索结果更相关。这还有助于减小索引的大小。
|
|
||||||
|
|
||||||
`REPO_INDEXER_EXCLUDE_VENDORED`(默认值为 true)将排除供应商文件不包含在索引中。
|
|
||||||
|
|
||||||
`REPO_INDEXER_INCLUDE`(默认值为空)是一个逗号分隔的 glob 模式列表,用于在索引中**包含**的文件。空列表表示“_包含所有文件_”。
|
|
||||||
`REPO_INDEXER_EXCLUDE`(默认值为空)是一个逗号分隔的 glob 模式列表,用于从索引中**排除**的文件。与该列表匹配的文件将不会被索引。`REPO_INDEXER_EXCLUDE` 优先于 `REPO_INDEXER_INCLUDE`。
|
|
||||||
|
|
||||||
模式匹配工作方式如下:
|
|
||||||
|
|
||||||
- 要匹配所有带有 `.txt` 扩展名的文件,无论在哪个目录中,请使用 `**.txt`。
|
|
||||||
- 要匹配仅在仓库的根级别中具有 `.txt` 扩展名的所有文件,请使用 `*.txt`。
|
|
||||||
- 要匹配 `resources/bin` 目录及其子目录中的所有文件,请使用 `resources/bin/**`。
|
|
||||||
- 要匹配位于 `resources/bin` 目录下的所有文件,请使用 `resources/bin/*`。
|
|
||||||
- 要匹配所有名为 `Makefile` 的文件,请使用 `**Makefile`。
|
|
||||||
- 匹配目录没有效果;模式 `resources/bin` 不会包含/排除该目录中的文件;`resources/bin/**` 会。
|
|
||||||
- 所有文件和模式都规范化为小写,因此 `**Makefile`、`**makefile` 和 `**MAKEFILE` 是等效的。
|
|
@ -1,389 +0,0 @@
|
|||||||
---
|
|
||||||
date: "2018-05-22T11:00:00+00:00"
|
|
||||||
title: "Reverse Proxies"
|
|
||||||
slug: "reverse-proxies"
|
|
||||||
sidebar_position: 16
|
|
||||||
toc: false
|
|
||||||
draft: false
|
|
||||||
aliases:
|
|
||||||
- /en-us/reverse-proxies
|
|
||||||
menu:
|
|
||||||
sidebar:
|
|
||||||
parent: "administration"
|
|
||||||
name: "Reverse Proxies"
|
|
||||||
sidebar_position: 16
|
|
||||||
identifier: "reverse-proxies"
|
|
||||||
---
|
|
||||||
|
|
||||||
# Reverse Proxies
|
|
||||||
|
|
||||||
## Nginx
|
|
||||||
|
|
||||||
If you want Nginx to serve your Gitea instance, add the following `server` section to the `http` section of `nginx.conf`:
|
|
||||||
|
|
||||||
```
|
|
||||||
server {
|
|
||||||
listen 80;
|
|
||||||
server_name git.example.com;
|
|
||||||
|
|
||||||
location / {
|
|
||||||
client_max_body_size 512M;
|
|
||||||
proxy_pass http://localhost:3000;
|
|
||||||
proxy_set_header Connection $http_connection;
|
|
||||||
proxy_set_header Upgrade $http_upgrade;
|
|
||||||
proxy_set_header Host $host;
|
|
||||||
proxy_set_header X-Real-IP $remote_addr;
|
|
||||||
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
|
||||||
proxy_set_header X-Forwarded-Proto $scheme;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Resolving Error: 413 Request Entity Too Large
|
|
||||||
|
|
||||||
This error indicates nginx is configured to restrict the file upload size,
|
|
||||||
it affects attachment uploading, form posting, package uploading and LFS pushing, etc.
|
|
||||||
You can fine tune the `client_max_body_size` option according to [nginx document](http://nginx.org/en/docs/http/ngx_http_core_module.html#client_max_body_size).
|
|
||||||
|
|
||||||
## Nginx with a sub-path
|
|
||||||
|
|
||||||
In case you already have a site, and you want Gitea to share the domain name, you can setup Nginx to serve Gitea under a sub-path by adding the following `server` section inside the `http` section of `nginx.conf`:
|
|
||||||
|
|
||||||
```
|
|
||||||
server {
|
|
||||||
listen 80;
|
|
||||||
server_name git.example.com;
|
|
||||||
|
|
||||||
# Note: Trailing slash
|
|
||||||
location /gitea/ {
|
|
||||||
client_max_body_size 512M;
|
|
||||||
|
|
||||||
# make nginx use unescaped URI, keep "%2F" as is
|
|
||||||
rewrite ^ $request_uri;
|
|
||||||
rewrite ^/gitea(/.*) $1 break;
|
|
||||||
proxy_pass http://127.0.0.1:3000$uri;
|
|
||||||
|
|
||||||
# other common HTTP headers, see the "Nginx" config section above
|
|
||||||
proxy_set_header ...
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Then you **MUST** set something like `[server] ROOT_URL = http://git.example.com/git/` correctly in your configuration.
|
|
||||||
|
|
||||||
## Nginx and serve static resources directly
|
|
||||||
|
|
||||||
We can tune the performance in splitting requests into categories static and dynamic.
|
|
||||||
|
|
||||||
CSS files, JavaScript files, images and web fonts are static content.
|
|
||||||
The front page, a repository view or issue list is dynamic content.
|
|
||||||
|
|
||||||
Nginx can serve static resources directly and proxy only the dynamic requests to Gitea.
|
|
||||||
Nginx is optimized for serving static content, while the proxying of large responses might be the opposite of that
|
|
||||||
(see [https://serverfault.com/q/587386](https://serverfault.com/q/587386)).
|
|
||||||
|
|
||||||
Download a snapshot of the Gitea source repository to `/path/to/gitea/`.
|
|
||||||
After this, run `make frontend` in the repository directory to generate the static resources. We are only interested in the `public/` directory for this task, so you can delete the rest.
|
|
||||||
(You will need to have [Node with npm](https://nodejs.org/en/download/) and `make` installed to generate the static resources)
|
|
||||||
|
|
||||||
Depending on the scale of your user base, you might want to split the traffic to two distinct servers,
|
|
||||||
or use a cdn for the static files.
|
|
||||||
|
|
||||||
### Single node and single domain
|
|
||||||
|
|
||||||
Set `[server] STATIC_URL_PREFIX = /_/static` in your configuration.
|
|
||||||
|
|
||||||
```apacheconf
|
|
||||||
server {
|
|
||||||
listen 80;
|
|
||||||
server_name git.example.com;
|
|
||||||
|
|
||||||
location /_/static/assets/ {
|
|
||||||
alias /path/to/gitea/public/;
|
|
||||||
}
|
|
||||||
|
|
||||||
location / {
|
|
||||||
proxy_pass http://localhost:3000;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### Two nodes and two domains
|
|
||||||
|
|
||||||
Set `[server] STATIC_URL_PREFIX = http://cdn.example.com/gitea` in your configuration.
|
|
||||||
|
|
||||||
```apacheconf
|
|
||||||
# application server running Gitea
|
|
||||||
server {
|
|
||||||
listen 80;
|
|
||||||
server_name git.example.com;
|
|
||||||
|
|
||||||
location / {
|
|
||||||
proxy_pass http://localhost:3000;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
```apacheconf
|
|
||||||
# static content delivery server
|
|
||||||
server {
|
|
||||||
listen 80;
|
|
||||||
server_name cdn.example.com;
|
|
||||||
|
|
||||||
location /gitea/ {
|
|
||||||
alias /path/to/gitea/public/;
|
|
||||||
}
|
|
||||||
|
|
||||||
location / {
|
|
||||||
return 404;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Apache HTTPD
|
|
||||||
|
|
||||||
If you want Apache HTTPD to serve your Gitea instance, you can add the following to your Apache HTTPD configuration (usually located at `/etc/apache2/httpd.conf` in Ubuntu):
|
|
||||||
|
|
||||||
```apacheconf
|
|
||||||
<VirtualHost *:80>
|
|
||||||
...
|
|
||||||
ProxyPreserveHost On
|
|
||||||
ProxyRequests off
|
|
||||||
AllowEncodedSlashes NoDecode
|
|
||||||
ProxyPass / http://localhost:3000/ nocanon
|
|
||||||
</VirtualHost>
|
|
||||||
```
|
|
||||||
|
|
||||||
Note: The following Apache HTTPD mods must be enabled: `proxy`, `proxy_http`.
|
|
||||||
|
|
||||||
If you wish to use Let's Encrypt with webroot validation, add the line `ProxyPass /.well-known !` before `ProxyPass` to disable proxying these requests to Gitea.
|
|
||||||
|
|
||||||
## Apache HTTPD with a sub-path
|
|
||||||
|
|
||||||
In case you already have a site, and you want Gitea to share the domain name, you can setup Apache HTTPD to serve Gitea under a sub-path by adding the following to you Apache HTTPD configuration (usually located at `/etc/apache2/httpd.conf` in Ubuntu):
|
|
||||||
|
|
||||||
```apacheconf
|
|
||||||
<VirtualHost *:80>
|
|
||||||
...
|
|
||||||
<Proxy *>
|
|
||||||
Order allow,deny
|
|
||||||
Allow from all
|
|
||||||
</Proxy>
|
|
||||||
AllowEncodedSlashes NoDecode
|
|
||||||
# Note: no trailing slash after either /git or port
|
|
||||||
ProxyPass /git http://localhost:3000 nocanon
|
|
||||||
</VirtualHost>
|
|
||||||
```
|
|
||||||
|
|
||||||
Then you **MUST** set something like `[server] ROOT_URL = http://git.example.com/git/` correctly in your configuration.
|
|
||||||
|
|
||||||
Note: The following Apache HTTPD mods must be enabled: `proxy`, `proxy_http`.
|
|
||||||
|
|
||||||
## Caddy
|
|
||||||
|
|
||||||
If you want Caddy to serve your Gitea instance, you can add the following server block to your Caddyfile:
|
|
||||||
|
|
||||||
```apacheconf
|
|
||||||
git.example.com {
|
|
||||||
reverse_proxy localhost:3000
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
## Caddy with a sub-path
|
|
||||||
|
|
||||||
In case you already have a site, and you want Gitea to share the domain name, you can setup Caddy to serve Gitea under a sub-path by adding the following to your server block in your Caddyfile:
|
|
||||||
|
|
||||||
```apacheconf
|
|
||||||
git.example.com {
|
|
||||||
route /git/* {
|
|
||||||
uri strip_prefix /git
|
|
||||||
reverse_proxy localhost:3000
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Then set `[server] ROOT_URL = http://git.example.com/git/` in your configuration.
|
|
||||||
|
|
||||||
## IIS
|
|
||||||
|
|
||||||
If you wish to run Gitea with IIS. You will need to setup IIS with URL Rewrite as reverse proxy.
|
|
||||||
|
|
||||||
1. Setup an empty website in IIS, named let's say, `Gitea Proxy`.
|
|
||||||
2. Follow the first two steps in [Microsoft's Technical Community Guide to Setup IIS with URL Rewrite](https://techcommunity.microsoft.com/t5/iis-support-blog/setup-iis-with-url-rewrite-as-a-reverse-proxy-for-real-world/ba-p/846222#M343). That is:
|
|
||||||
|
|
||||||
- Install Application Request Routing (ARR for short) either by using the Microsoft Web Platform Installer 5.1 (WebPI) or downloading the extension from [IIS.net](https://www.iis.net/downloads/microsoft/application-request-routing)
|
|
||||||
- Once the module is installed in IIS, you will see a new Icon in the IIS Administration Console called URL Rewrite.
|
|
||||||
- Open the IIS Manager Console and click on the `Gitea Proxy` Website from the tree view on the left. Select and double click the URL Rewrite Icon from the middle pane to load the URL Rewrite interface.
|
|
||||||
- Choose the `Add Rule` action from the right pane of the management console and select the `Reverse Proxy Rule` from the `Inbound and Outbound Rules` category.
|
|
||||||
- In the Inbound Rules section, set the server name to be the host that Gitea is running on with its port. e.g. if you are running Gitea on the localhost with port 3000, the following should work: `127.0.0.1:3000`
|
|
||||||
- Enable SSL Offloading
|
|
||||||
- In the Outbound Rules, ensure `Rewrite the domain names of the links in HTTP response` is set and set the `From:` field as above and the `To:` to your external hostname, say: `git.example.com`
|
|
||||||
- Now edit the `web.config` for your website to match the following: (changing `127.0.0.1:3000` and `git.example.com` as appropriate)
|
|
||||||
|
|
||||||
```xml
|
|
||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<configuration>
|
|
||||||
<system.web>
|
|
||||||
<httpRuntime requestPathInvalidCharacters="" />
|
|
||||||
</system.web>
|
|
||||||
<system.webServer>
|
|
||||||
<security>
|
|
||||||
<requestFiltering>
|
|
||||||
<hiddenSegments>
|
|
||||||
<clear />
|
|
||||||
</hiddenSegments>
|
|
||||||
<denyUrlSequences>
|
|
||||||
<clear />
|
|
||||||
</denyUrlSequences>
|
|
||||||
<fileExtensions allowUnlisted="true">
|
|
||||||
<clear />
|
|
||||||
</fileExtensions>
|
|
||||||
</requestFiltering>
|
|
||||||
</security>
|
|
||||||
<rewrite>
|
|
||||||
<rules useOriginalURLEncoding="false">
|
|
||||||
<rule name="ReverseProxyInboundRule1" stopProcessing="true">
|
|
||||||
<match url="(.*)" />
|
|
||||||
<action type="Rewrite" url="http://127.0.0.1:3000{UNENCODED_URL}" />
|
|
||||||
<serverVariables>
|
|
||||||
<set name="HTTP_X_ORIGINAL_ACCEPT_ENCODING" value="HTTP_ACCEPT_ENCODING" />
|
|
||||||
<set name="HTTP_ACCEPT_ENCODING" value="" />
|
|
||||||
</serverVariables>
|
|
||||||
</rule>
|
|
||||||
</rules>
|
|
||||||
<outboundRules>
|
|
||||||
<rule name="ReverseProxyOutboundRule1" preCondition="ResponseIsHtml1">
|
|
||||||
<!-- set the pattern correctly here - if you only want to accept http or https -->
|
|
||||||
<!-- change the pattern and the action value as appropriate -->
|
|
||||||
<match filterByTags="A, Form, Img" pattern="^http(s)?://127.0.0.1:3000/(.*)" />
|
|
||||||
<action type="Rewrite" value="http{R:1}://git.example.com/{R:2}" />
|
|
||||||
</rule>
|
|
||||||
<rule name="RestoreAcceptEncoding" preCondition="NeedsRestoringAcceptEncoding">
|
|
||||||
<match serverVariable="HTTP_ACCEPT_ENCODING" pattern="^(.*)" />
|
|
||||||
<action type="Rewrite" value="{HTTP_X_ORIGINAL_ACCEPT_ENCODING}" />
|
|
||||||
</rule>
|
|
||||||
<preConditions>
|
|
||||||
<preCondition name="ResponseIsHtml1">
|
|
||||||
<add input="{RESPONSE_CONTENT_TYPE}" pattern="^text/html" />
|
|
||||||
</preCondition>
|
|
||||||
<preCondition name="NeedsRestoringAcceptEncoding">
|
|
||||||
<add input="{HTTP_X_ORIGINAL_ACCEPT_ENCODING}" pattern=".+" />
|
|
||||||
</preCondition>
|
|
||||||
</preConditions>
|
|
||||||
</outboundRules>
|
|
||||||
</rewrite>
|
|
||||||
<urlCompression doDynamicCompression="true" />
|
|
||||||
<handlers>
|
|
||||||
<clear />
|
|
||||||
<add name="StaticFile" path="*" verb="*" modules="StaticFileModule,DefaultDocumentModule,DirectoryListingModule" resourceType="Either" requireAccess="Read" />
|
|
||||||
</handlers>
|
|
||||||
<!-- Map all extensions to the same MIME type, so all files can be
|
|
||||||
downloaded. -->
|
|
||||||
<staticContent>
|
|
||||||
<clear />
|
|
||||||
<mimeMap fileExtension="*" mimeType="application/octet-stream" />
|
|
||||||
</staticContent>
|
|
||||||
</system.webServer>
|
|
||||||
</configuration>
|
|
||||||
```
|
|
||||||
|
|
||||||
## HAProxy
|
|
||||||
|
|
||||||
If you want HAProxy to serve your Gitea instance, you can add the following to your HAProxy configuration
|
|
||||||
|
|
||||||
add an acl in the frontend section to redirect calls to gitea.example.com to the correct backend
|
|
||||||
|
|
||||||
```
|
|
||||||
frontend http-in
|
|
||||||
...
|
|
||||||
acl acl_gitea hdr(host) -i gitea.example.com
|
|
||||||
use_backend gitea if acl_gitea
|
|
||||||
...
|
|
||||||
```
|
|
||||||
|
|
||||||
add the previously defined backend section
|
|
||||||
|
|
||||||
```
|
|
||||||
backend gitea
|
|
||||||
server localhost:3000 check
|
|
||||||
```
|
|
||||||
|
|
||||||
If you redirect the http content to https, the configuration work the same way, just remember that the connection between HAProxy and Gitea will be done via http so you do not have to enable https in Gitea's configuration.
|
|
||||||
|
|
||||||
## HAProxy with a sub-path
|
|
||||||
|
|
||||||
In case you already have a site, and you want Gitea to share the domain name, you can setup HAProxy to serve Gitea under a sub-path by adding the following to you HAProxy configuration:
|
|
||||||
|
|
||||||
```
|
|
||||||
frontend http-in
|
|
||||||
...
|
|
||||||
acl acl_gitea path_beg /gitea
|
|
||||||
use_backend gitea if acl_gitea
|
|
||||||
...
|
|
||||||
```
|
|
||||||
|
|
||||||
With that configuration http://example.com/gitea/ will redirect to your Gitea instance.
|
|
||||||
|
|
||||||
then for the backend section
|
|
||||||
|
|
||||||
```
|
|
||||||
backend gitea
|
|
||||||
http-request replace-path /gitea\/?(.*) \/\1
|
|
||||||
server localhost:3000 check
|
|
||||||
```
|
|
||||||
|
|
||||||
The added http-request will automatically add a trailing slash if needed and internally remove /gitea from the path to allow it to work correctly with Gitea by setting properly http://example.com/gitea as the root.
|
|
||||||
|
|
||||||
Then you **MUST** set something like `[server] ROOT_URL = http://example.com/gitea/` correctly in your configuration.
|
|
||||||
|
|
||||||
## Traefik
|
|
||||||
|
|
||||||
If you want traefik to serve your Gitea instance, you can add the following label section to your `docker-compose.yaml` (Assuming the provider is docker).
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
gitea:
|
|
||||||
image: gitea/gitea
|
|
||||||
...
|
|
||||||
labels:
|
|
||||||
- "traefik.enable=true"
|
|
||||||
- "traefik.http.routers.gitea.rule=Host(`example.com`)"
|
|
||||||
- "traefik.http.services.gitea-websecure.loadbalancer.server.port=3000"
|
|
||||||
```
|
|
||||||
|
|
||||||
This config assumes that you are handling HTTPS on the traefik side and using HTTP between Gitea and traefik.
|
|
||||||
|
|
||||||
## Traefik with a sub-path
|
|
||||||
|
|
||||||
In case you already have a site, and you want Gitea to share the domain name, you can setup Traefik to serve Gitea under a sub-path by adding the following to your `docker-compose.yaml` (Assuming the provider is docker) :
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
gitea:
|
|
||||||
image: gitea/gitea
|
|
||||||
...
|
|
||||||
labels:
|
|
||||||
- "traefik.enable=true"
|
|
||||||
- "traefik.http.routers.gitea.rule=Host(`example.com`) && PathPrefix(`/gitea`)"
|
|
||||||
- "traefik.http.services.gitea-websecure.loadbalancer.server.port=3000"
|
|
||||||
- "traefik.http.middlewares.gitea-stripprefix.stripprefix.prefixes=/gitea"
|
|
||||||
- "traefik.http.routers.gitea.middlewares=gitea-stripprefix"
|
|
||||||
```
|
|
||||||
|
|
||||||
This config assumes that you are handling HTTPS on the traefik side and using HTTP between Gitea and traefik.
|
|
||||||
|
|
||||||
Then you **MUST** set something like `[server] ROOT_URL = http://example.com/gitea/` correctly in your configuration.
|
|
||||||
|
|
||||||
## General sub-path configuration
|
|
||||||
|
|
||||||
Usually it's not recommended to put Gitea in a sub-path, it's not widely used and may have some issues in rare cases.
|
|
||||||
|
|
||||||
If you really need to do so, to make Gitea works with sub-path (eg: `http://example.com/gitea/`), here are the requirements:
|
|
||||||
|
|
||||||
1. Set `[server] ROOT_URL = http://example.com/gitea/` in your `app.ini` file.
|
|
||||||
2. Make the reverse-proxy pass `http://example.com/gitea/foo` to `http://gitea-server:3000/foo`.
|
|
||||||
3. Make sure the reverse-proxy not decode the URI, the request `http://example.com/gitea/a%2Fb` should be passed as `http://gitea-server:3000/a%2Fb`.
|
|
||||||
|
|
||||||
## Docker / Container Registry
|
|
||||||
|
|
||||||
The container registry uses a fixed sub-path `/v2` which can't be changed.
|
|
||||||
Even if you deploy Gitea with a different sub-path, `/v2` will be used by the `docker` client.
|
|
||||||
Therefore you may need to add an additional route to your reverse proxy configuration.
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user