2023-03-01 04:26:13 +00:00
/ *
* Vencord , a modification for Discord ' s desktop app
* Copyright ( c ) 2022 Vendicated and contributors
*
* This program is free software : you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation , either version 3 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program . If not , see < https : //www.gnu.org/licenses/>.
* /
import { showNotification } from "@api/Notifications" ;
2023-05-05 23:36:00 +00:00
import { definePluginSettings } from "@api/Settings" ;
2023-03-01 04:26:13 +00:00
import { Devs } from "@utils/constants" ;
2023-05-05 23:36:00 +00:00
import { Logger } from "@utils/Logger" ;
2023-03-01 04:26:13 +00:00
import { closeAllModals } from "@utils/modal" ;
import definePlugin , { OptionType } from "@utils/types" ;
import { maybePromptToUpdate } from "@utils/updater" ;
2023-11-30 03:43:23 +00:00
import { filters , findBulk , proxyLazyWebpack } from "@webpack" ;
2023-11-29 19:14:05 +00:00
import { FluxDispatcher , NavigationRouter , SelectedChannelStore } from "@webpack/common" ;
2023-03-01 04:26:13 +00:00
import type { ReactElement } from "react" ;
const CrashHandlerLogger = new Logger ( "CrashHandler" ) ;
2023-11-30 03:43:23 +00:00
const { ModalStack , DraftManager , DraftType , closeExpressionPicker } = proxyLazyWebpack ( ( ) = > {
const modules = findBulk (
filters . byProps ( "pushLazy" , "popAll" ) ,
filters . byProps ( "clearDraft" , "saveDraft" ) ,
filters . byProps ( "DraftType" ) ,
filters . byProps ( "closeExpressionPicker" , "openExpressionPicker" ) ,
) ;
return {
ModalStack : modules [ 0 ] ,
DraftManager : modules [ 1 ] ,
DraftType : modules [ 2 ] ? . DraftType ,
closeExpressionPicker : modules [ 3 ] ? . closeExpressionPicker ,
} ;
} ) ;
2023-03-01 04:26:13 +00:00
const settings = definePluginSettings ( {
attemptToPreventCrashes : {
type : OptionType . BOOLEAN ,
description : "Whether to attempt to prevent Discord crashes." ,
default : true
} ,
attemptToNavigateToHome : {
type : OptionType . BOOLEAN ,
description : "Whether to attempt to navigate to the home when preventing Discord crashes." ,
default : false
}
} ) ;
2023-03-06 21:54:01 +00:00
let crashCount : number = 0 ;
2023-03-21 06:07:16 +00:00
let lastCrashTimestamp : number = 0 ;
2023-04-05 03:06:04 +00:00
let shouldAttemptNextHandle = false ;
2023-03-06 21:54:01 +00:00
2023-03-01 04:26:13 +00:00
export default definePlugin ( {
name : "CrashHandler" ,
description : "Utility plugin for handling and possibly recovering from Crashes without a restart" ,
authors : [ Devs . Nuckyz ] ,
enabledByDefault : true ,
settings ,
patches : [
{
find : ".Messages.ERRORS_UNEXPECTED_CRASH" ,
replacement : {
match : /(?=this\.setState\()/ ,
replace : "$self.handleCrash(this)||"
}
}
] ,
handleCrash ( _this : ReactElement & { forceUpdate : ( ) = > void ; } ) {
2023-04-05 03:06:04 +00:00
if ( Date . now ( ) - lastCrashTimestamp <= 1 _000 && ! shouldAttemptNextHandle ) return true ;
shouldAttemptNextHandle = false ;
2023-03-06 21:54:01 +00:00
if ( ++ crashCount > 5 ) {
try {
showNotification ( {
color : "#eed202" ,
title : "Discord has crashed!" ,
body : "Awn :( Discord has crashed more than five times, not attempting to recover." ,
2023-04-01 00:47:49 +00:00
noPersist : true ,
2023-03-06 21:54:01 +00:00
} ) ;
} catch { }
2023-03-21 06:07:16 +00:00
lastCrashTimestamp = Date . now ( ) ;
2023-03-06 21:54:01 +00:00
return false ;
}
setTimeout ( ( ) = > crashCount -- , 60 _000 ) ;
2023-03-01 04:26:13 +00:00
try {
2023-03-06 21:54:01 +00:00
if ( crashCount === 1 ) maybePromptToUpdate ( "Uh oh, Discord has just crashed... but good news, there is a Vencord update available that might fix this issue! Would you like to update now?" , true ) ;
2023-03-01 04:26:13 +00:00
if ( settings . store . attemptToPreventCrashes ) {
this . handlePreventCrash ( _this ) ;
return true ;
}
return false ;
} catch ( err ) {
CrashHandlerLogger . error ( "Failed to handle crash" , err ) ;
2023-03-06 21:54:01 +00:00
return false ;
2023-03-21 06:07:16 +00:00
} finally {
lastCrashTimestamp = Date . now ( ) ;
2023-03-01 04:26:13 +00:00
}
} ,
handlePreventCrash ( _this : ReactElement & { forceUpdate : ( ) = > void ; } ) {
2023-03-21 06:07:16 +00:00
if ( Date . now ( ) - lastCrashTimestamp >= 1 _000 ) {
try {
showNotification ( {
color : "#eed202" ,
title : "Discord has crashed!" ,
body : "Attempting to recover..." ,
2023-04-01 00:47:49 +00:00
noPersist : true ,
2023-03-21 06:07:16 +00:00
} ) ;
} catch { }
}
2023-03-01 04:26:13 +00:00
2023-11-29 19:14:05 +00:00
try {
const channelId = SelectedChannelStore . getChannelId ( ) ;
DraftManager . clearDraft ( channelId , DraftType . ChannelMessage ) ;
DraftManager . clearDraft ( channelId , DraftType . FirstThreadMessage ) ;
} catch ( err ) {
CrashHandlerLogger . debug ( "Failed to clear drafts." , err ) ;
}
try {
closeExpressionPicker ( ) ;
}
catch ( err ) {
CrashHandlerLogger . debug ( "Failed to close expression picker." , err ) ;
}
2023-03-01 04:26:13 +00:00
try {
FluxDispatcher . dispatch ( { type : "CONTEXT_MENU_CLOSE" } ) ;
} catch ( err ) {
CrashHandlerLogger . debug ( "Failed to close open context menu." , err ) ;
}
try {
2023-11-30 03:43:23 +00:00
ModalStack . popAll ( ) ;
2023-03-01 04:26:13 +00:00
} catch ( err ) {
CrashHandlerLogger . debug ( "Failed to close old modals." , err ) ;
}
try {
closeAllModals ( ) ;
} catch ( err ) {
CrashHandlerLogger . debug ( "Failed to close all open modals." , err ) ;
}
try {
FluxDispatcher . dispatch ( { type : "USER_PROFILE_MODAL_CLOSE" } ) ;
} catch ( err ) {
CrashHandlerLogger . debug ( "Failed to close user popout." , err ) ;
}
try {
FluxDispatcher . dispatch ( { type : "LAYER_POP_ALL" } ) ;
} catch ( err ) {
CrashHandlerLogger . debug ( "Failed to pop all layers." , err ) ;
}
if ( settings . store . attemptToNavigateToHome ) {
try {
NavigationRouter . transitionTo ( "/channels/@me" ) ;
} catch ( err ) {
CrashHandlerLogger . debug ( "Failed to navigate to home" , err ) ;
}
}
try {
2023-04-05 03:06:04 +00:00
shouldAttemptNextHandle = true ;
2023-03-01 04:26:13 +00:00
_this . forceUpdate ( ) ;
} catch ( err ) {
CrashHandlerLogger . debug ( "Failed to update crash handler component." , err ) ;
}
}
} ) ;