Fix 404 and 410 API errors being silently discarded in WebUI (#13279)
* Fix 404 and 410 API errors being silently discarded in WebUI Fixes #13278 * Return more appropriate error when user replies to a deleted toot * Please CodeClimate * Fix 404/410 errors on fetching account timelines & identity proofs * Refactor error handling * Move error message string to statuses.errors
This commit is contained in:
		
					parent
					
						
							
								7ddbbdea6d
							
						
					
				
			
			
				commit
				
					
						0d117c106a
					
				
			
		
					 7 changed files with 18 additions and 4 deletions
				
			
		|  | @ -7,6 +7,7 @@ class Api::V1::StatusesController < Api::BaseController | ||||||
|   before_action -> { doorkeeper_authorize! :write, :'write:statuses' }, only:   [:create, :destroy] |   before_action -> { doorkeeper_authorize! :write, :'write:statuses' }, only:   [:create, :destroy] | ||||||
|   before_action :require_user!, except:  [:show, :context] |   before_action :require_user!, except:  [:show, :context] | ||||||
|   before_action :set_status, only:       [:show, :context] |   before_action :set_status, only:       [:show, :context] | ||||||
|  |   before_action :set_thread, only:       [:create] | ||||||
| 
 | 
 | ||||||
|   override_rate_limit_headers :create, family: :statuses |   override_rate_limit_headers :create, family: :statuses | ||||||
| 
 | 
 | ||||||
|  | @ -36,7 +37,7 @@ class Api::V1::StatusesController < Api::BaseController | ||||||
|   def create |   def create | ||||||
|     @status = PostStatusService.new.call(current_user.account, |     @status = PostStatusService.new.call(current_user.account, | ||||||
|                                          text: status_params[:status], |                                          text: status_params[:status], | ||||||
|                                          thread: status_params[:in_reply_to_id].blank? ? nil : Status.find(status_params[:in_reply_to_id]), |                                          thread: @thread, | ||||||
|                                          media_ids: status_params[:media_ids], |                                          media_ids: status_params[:media_ids], | ||||||
|                                          sensitive: status_params[:sensitive], |                                          sensitive: status_params[:sensitive], | ||||||
|                                          spoiler_text: status_params[:spoiler_text], |                                          spoiler_text: status_params[:spoiler_text], | ||||||
|  | @ -69,6 +70,12 @@ class Api::V1::StatusesController < Api::BaseController | ||||||
|     raise ActiveRecord::RecordNotFound |     raise ActiveRecord::RecordNotFound | ||||||
|   end |   end | ||||||
| 
 | 
 | ||||||
|  |   def set_thread | ||||||
|  |     @thread = status_params[:in_reply_to_id].blank? ? nil : Status.find(status_params[:in_reply_to_id]) | ||||||
|  |   rescue ActiveRecord::RecordNotFound | ||||||
|  |     render json: { error: I18n.t('statuses.errors.in_reply_not_found') }, status: 404 | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|   def status_params |   def status_params | ||||||
|     params.permit( |     params.permit( | ||||||
|       :status, |       :status, | ||||||
|  |  | ||||||
|  | @ -396,6 +396,7 @@ export function fetchFollowersFail(id, error) { | ||||||
|     type: FOLLOWERS_FETCH_FAIL, |     type: FOLLOWERS_FETCH_FAIL, | ||||||
|     id, |     id, | ||||||
|     error, |     error, | ||||||
|  |     skipNotFound: true, | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | @ -482,6 +483,7 @@ export function fetchFollowingFail(id, error) { | ||||||
|     type: FOLLOWING_FETCH_FAIL, |     type: FOLLOWING_FETCH_FAIL, | ||||||
|     id, |     id, | ||||||
|     error, |     error, | ||||||
|  |     skipNotFound: true, | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | @ -571,6 +573,7 @@ export function fetchRelationshipsFail(error) { | ||||||
|     type: RELATIONSHIPS_FETCH_FAIL, |     type: RELATIONSHIPS_FETCH_FAIL, | ||||||
|     error, |     error, | ||||||
|     skipLoading: true, |     skipLoading: true, | ||||||
|  |     skipNotFound: true, | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -34,11 +34,11 @@ export function showAlert(title = messages.unexpectedTitle, message = messages.u | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| export function showAlertForError(error) { | export function showAlertForError(error, skipNotFound = false) { | ||||||
|   if (error.response) { |   if (error.response) { | ||||||
|     const { data, status, statusText, headers } = error.response; |     const { data, status, statusText, headers } = error.response; | ||||||
| 
 | 
 | ||||||
|     if (status === 404 || status === 410) { |     if (skipNotFound && (status === 404 || status === 410)) { | ||||||
|       // Skip these errors as they are reflected in the UI
 |       // Skip these errors as they are reflected in the UI
 | ||||||
|       return { type: ALERT_NOOP }; |       return { type: ALERT_NOOP }; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -27,4 +27,5 @@ export const fetchAccountIdentityProofsFail = (accountId, err) => ({ | ||||||
|   type: IDENTITY_PROOFS_ACCOUNT_FETCH_FAIL, |   type: IDENTITY_PROOFS_ACCOUNT_FETCH_FAIL, | ||||||
|   accountId, |   accountId, | ||||||
|   err, |   err, | ||||||
|  |   skipNotFound: true, | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  | @ -149,6 +149,7 @@ export function expandTimelineFail(timeline, error, isLoadingMore) { | ||||||
|     timeline, |     timeline, | ||||||
|     error, |     error, | ||||||
|     skipLoading: !isLoadingMore, |     skipLoading: !isLoadingMore, | ||||||
|  |     skipNotFound: timeline.startsWith('account:'), | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -8,7 +8,7 @@ export default function errorsMiddleware() { | ||||||
|       const isFail = new RegExp(`${defaultFailSuffix}$`, 'g'); |       const isFail = new RegExp(`${defaultFailSuffix}$`, 'g'); | ||||||
| 
 | 
 | ||||||
|       if (action.type.match(isFail)) { |       if (action.type.match(isFail)) { | ||||||
|         dispatch(showAlertForError(action.error)); |         dispatch(showAlertForError(action.error, action.skipNotFound)); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1078,6 +1078,8 @@ en: | ||||||
|     disallowed_hashtags: |     disallowed_hashtags: | ||||||
|       one: 'contained a disallowed hashtag: %{tags}' |       one: 'contained a disallowed hashtag: %{tags}' | ||||||
|       other: 'contained the disallowed hashtags: %{tags}' |       other: 'contained the disallowed hashtags: %{tags}' | ||||||
|  |     errors: | ||||||
|  |       in_reply_not_found: The status you are trying to reply to does not appear to exist. | ||||||
|     language_detection: Automatically detect language |     language_detection: Automatically detect language | ||||||
|     open_in_web: Open in web |     open_in_web: Open in web | ||||||
|     over_character_limit: character limit of %{max} exceeded |     over_character_limit: character limit of %{max} exceeded | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue