Skip to content

Commit e4e3df6

Browse files
Gustedlunnywxiaoguang
authored
Handle invalid issues (#18111)
* Handle invalid issues - When you hover over a issue reference, and the issue doesn't exist, it will just hang on the loading animation. - This patch fixes that by showing them the pop-up with a "Error occured" message. * Add I18N * refactor * fix comment for lint * fix unit test for i18n * fix unit test for i18n * add comments Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com> Co-authored-by: wxiaoguang <wxiaoguang@gmail.com>
1 parent d2fac63 commit e4e3df6

File tree

6 files changed

+45
-23
lines changed

6 files changed

+45
-23
lines changed

integrations/api_releases_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -207,7 +207,7 @@ func TestAPIGetReleaseByTag(t *testing.T) {
207207

208208
var err *api.APIError
209209
DecodeJSON(t, resp, &err)
210-
assert.EqualValues(t, "Not Found", err.Message)
210+
assert.NotEmpty(t, err.Message)
211211
}
212212

213213
func TestAPIDeleteReleaseByTagName(t *testing.T) {

integrations/repo_branch_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -141,7 +141,7 @@ func TestCreateBranchInvalidCSRF(t *testing.T) {
141141
resp = session.MakeRequest(t, NewRequest(t, "GET", loc), http.StatusOK)
142142
htmlDoc := NewHTMLParser(t, resp.Body)
143143
assert.Equal(t,
144-
"Bad Request: Invalid CSRF token",
144+
"Bad Request: invalid CSRF token",
145145
strings.TrimSpace(htmlDoc.doc.Find(".ui.message").Text()),
146146
)
147147
}

modules/context/api.go

+14-9
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,11 @@ type APIContext struct {
3030
Org *APIOrganization
3131
}
3232

33+
// Currently, we have the following common fields in error response:
34+
// * message: the message for end users (it shouldn't be used for error type detection)
35+
// if we need to indicate some errors, we should introduce some new fields like ErrorCode or ErrorType
36+
// * url: the swagger document URL
37+
3338
// APIError is error format response
3439
// swagger:response error
3540
type APIError struct {
@@ -47,8 +52,8 @@ type APIValidationError struct {
4752
// APIInvalidTopicsError is error format response to invalid topics
4853
// swagger:response invalidTopicsError
4954
type APIInvalidTopicsError struct {
50-
Topics []string `json:"invalidTopics"`
51-
Message string `json:"message"`
55+
Message string `json:"message"`
56+
InvalidTopics []string `json:"invalidTopics"`
5257
}
5358

5459
//APIEmpty is an empty response
@@ -122,9 +127,9 @@ func (ctx *APIContext) InternalServerError(err error) {
122127
})
123128
}
124129

125-
var (
126-
apiContextKey interface{} = "default_api_context"
127-
)
130+
type apiContextKeyType struct{}
131+
132+
var apiContextKey = apiContextKeyType{}
128133

129134
// WithAPIContext set up api context in request
130135
func WithAPIContext(req *http.Request, ctx *APIContext) *http.Request {
@@ -351,7 +356,7 @@ func ReferencesGitRepo(allowEmpty bool) func(http.Handler) http.Handler {
351356
// NotFound handles 404s for APIContext
352357
// String will replace message, errors will be added to a slice
353358
func (ctx *APIContext) NotFound(objs ...interface{}) {
354-
var message = "Not Found"
359+
var message = ctx.Tr("error.not_found")
355360
var errors []string
356361
for _, obj := range objs {
357362
// Ignore nil
@@ -367,9 +372,9 @@ func (ctx *APIContext) NotFound(objs ...interface{}) {
367372
}
368373

369374
ctx.JSON(http.StatusNotFound, map[string]interface{}{
370-
"message": message,
371-
"documentation_url": setting.API.SwaggerURL,
372-
"errors": errors,
375+
"message": message,
376+
"url": setting.API.SwaggerURL,
377+
"errors": errors,
373378
})
374379
}
375380

options/locale/locale_en-US.ini

+5-3
Original file line numberDiff line numberDiff line change
@@ -104,10 +104,12 @@ error404 = The page you are trying to reach either <strong>does not exist</stron
104104
never = Never
105105

106106
[error]
107-
occurred = An error has occurred
108-
report_message = If you are sure this is a Gitea bug, please search for issue on <a href="https://github.com/go-gitea/gitea/issues">GitHub</a> and open new issue if necessary.
107+
occurred = An error occurred
108+
report_message = If you are sure this is a Gitea bug, please search for issues on <a href="https://github.com/go-gitea/gitea/issues" target="_blank">GitHub</a> or open a new issue if necessary.
109109
missing_csrf = Bad Request: no CSRF token present
110-
invalid_csrf = Bad Request: Invalid CSRF token
110+
invalid_csrf = Bad Request: invalid CSRF token
111+
not_found = The target couldn't be found.
112+
network_error = Network error
111113
112114
[startpage]
113115
app_desc = A painless, self-hosted Git service

templates/base/head.tmpl

+4-1
Original file line numberDiff line numberDiff line change
@@ -46,10 +46,13 @@
4646
]).values()),
4747
{{end}}
4848
mermaidMaxSourceCharacters: {{MermaidMaxSourceCharacters}},
49+
{{/* this global i18n object should only contain gereral texts. for specalized texts, it should be provied inside the related modules by: (1) API response (2) HTML data-attribute (3) PageData */}}
4950
i18n: {
5051
copy_success: '{{.i18n.Tr "copy_success"}}',
5152
copy_error: '{{.i18n.Tr "copy_error"}}',
52-
}
53+
error_occurred: '{{.i18n.Tr "error.occurred"}}',
54+
network_error: '{{.i18n.Tr "error.network_error"}}',
55+
},
5356
};
5457
{{/* in case some pages don't render the pageData, we make sure it is an object to prevent null access */}}
5558
window.config.pageData = window.config.pageData || {};

web_src/js/components/ContextPopup.vue

+20-8
Original file line numberDiff line numberDiff line change
@@ -16,13 +16,17 @@
1616
</div>
1717
</div>
1818
</div>
19+
<div v-if="!loading && issue === null">
20+
<p><small>{{ i18nErrorOccurred }}</small></p>
21+
<p>{{ i18nErrorMessage }}</p>
22+
</div>
1923
</div>
2024
</template>
2125

2226
<script>
2327
import {SvgIcon} from '../svg.js';
2428
25-
const {appSubUrl} = window.config;
29+
const {appSubUrl, i18n} = window.config;
2630
2731
// NOTE: see models/issue_label.go for similar implementation
2832
const srgbToLinear = (color) => {
@@ -49,7 +53,9 @@ export default {
4953
5054
data: () => ({
5155
loading: false,
52-
issue: null
56+
issue: null,
57+
i18nErrorOccurred: i18n.error_occurred,
58+
i18nErrorMessage: null,
5359
}),
5460
5561
computed: {
@@ -112,14 +118,20 @@ export default {
112118
methods: {
113119
load(data, callback) {
114120
this.loading = true;
115-
$.get(`${appSubUrl}/api/v1/repos/${data.owner}/${data.repo}/issues/${data.index}`, (issue) => {
121+
this.i18nErrorMessage = null;
122+
$.get(`${appSubUrl}/api/v1/repos/${data.owner}/${data.repo}/issues/${data.index}`).done((issue) => {
116123
this.issue = issue;
124+
}).fail((jqXHR) => {
125+
if (jqXHR.responseJSON && jqXHR.responseJSON.message) {
126+
this.i18nErrorMessage = jqXHR.responseJSON.message;
127+
} else {
128+
this.i18nErrorMessage = i18n.network_error;
129+
}
130+
}).always(() => {
117131
this.loading = false;
118-
this.$nextTick(() => {
119-
if (callback) {
120-
callback();
121-
}
122-
});
132+
if (callback) {
133+
this.$nextTick(callback);
134+
}
123135
});
124136
}
125137
}

0 commit comments

Comments
 (0)