mirror of
https://github.com/Sonarr/Sonarr.git
synced 2026-04-18 21:35:27 -04:00
Compare commits
1 Commits
log-files-
...
sidebar-cl
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9484904f60 |
@@ -1,5 +1,5 @@
|
|||||||
import React, { useCallback, useEffect } from 'react';
|
import React, { useCallback, useEffect } from 'react';
|
||||||
import { useSelector } from 'react-redux';
|
import { useDispatch, useSelector } from 'react-redux';
|
||||||
import Button from 'Components/Link/Button';
|
import Button from 'Components/Link/Button';
|
||||||
import LoadingIndicator from 'Components/Loading/LoadingIndicator';
|
import LoadingIndicator from 'Components/Loading/LoadingIndicator';
|
||||||
import InlineMarkdown from 'Components/Markdown/InlineMarkdown';
|
import InlineMarkdown from 'Components/Markdown/InlineMarkdown';
|
||||||
@@ -9,6 +9,7 @@ import ModalFooter from 'Components/Modal/ModalFooter';
|
|||||||
import ModalHeader from 'Components/Modal/ModalHeader';
|
import ModalHeader from 'Components/Modal/ModalHeader';
|
||||||
import usePrevious from 'Helpers/Hooks/usePrevious';
|
import usePrevious from 'Helpers/Hooks/usePrevious';
|
||||||
import { kinds } from 'Helpers/Props';
|
import { kinds } from 'Helpers/Props';
|
||||||
|
import { fetchUpdates } from 'Store/Actions/systemActions';
|
||||||
import UpdateChanges from 'System/Updates/UpdateChanges';
|
import UpdateChanges from 'System/Updates/UpdateChanges';
|
||||||
import useUpdates from 'System/Updates/useUpdates';
|
import useUpdates from 'System/Updates/useUpdates';
|
||||||
import Update from 'typings/Update';
|
import Update from 'typings/Update';
|
||||||
@@ -63,8 +64,9 @@ interface AppUpdatedModalContentProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function AppUpdatedModalContent(props: AppUpdatedModalContentProps) {
|
function AppUpdatedModalContent(props: AppUpdatedModalContentProps) {
|
||||||
|
const dispatch = useDispatch();
|
||||||
const { version, prevVersion } = useSelector((state: AppState) => state.app);
|
const { version, prevVersion } = useSelector((state: AppState) => state.app);
|
||||||
const { isFetched, error, data, refetch } = useUpdates();
|
const { isFetched, error, data } = useUpdates();
|
||||||
const previousVersion = usePrevious(version);
|
const previousVersion = usePrevious(version);
|
||||||
|
|
||||||
const { onModalClose } = props;
|
const { onModalClose } = props;
|
||||||
@@ -75,11 +77,15 @@ function AppUpdatedModalContent(props: AppUpdatedModalContentProps) {
|
|||||||
window.location.href = `${window.Sonarr.urlBase}/system/updates`;
|
window.location.href = `${window.Sonarr.urlBase}/system/updates`;
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
dispatch(fetchUpdates());
|
||||||
|
}, [dispatch]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (version !== previousVersion) {
|
if (version !== previousVersion) {
|
||||||
refetch();
|
dispatch(fetchUpdates());
|
||||||
}
|
}
|
||||||
}, [version, previousVersion, refetch]);
|
}, [version, previousVersion, dispatch]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ModalContent onModalClose={onModalClose}>
|
<ModalContent onModalClose={onModalClose}>
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
.header {
|
.header {
|
||||||
z-index: 3;
|
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
flex: 0 0 auto;
|
flex: 0 0 auto;
|
||||||
|
|||||||
@@ -7,6 +7,40 @@
|
|||||||
transform: translateX(0);
|
transform: translateX(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.sidebarHeader {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
height: $headerHeight;
|
||||||
|
}
|
||||||
|
|
||||||
|
.logoContainer {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding-left: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.logoLink {
|
||||||
|
line-height: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.logo {
|
||||||
|
width: 32px;
|
||||||
|
height: 32px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sidebarCloseButton {
|
||||||
|
composes: button from '~Components/Link/IconButton.css';
|
||||||
|
|
||||||
|
margin-right: 15px;
|
||||||
|
color: #e1e2e3;
|
||||||
|
text-align: center;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
color: var(--sonarrBlue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.sidebar {
|
.sidebar {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
|||||||
@@ -1,8 +1,13 @@
|
|||||||
// This file is automatically generated.
|
// This file is automatically generated.
|
||||||
// Please do not change this file!
|
// Please do not change this file!
|
||||||
interface CssExports {
|
interface CssExports {
|
||||||
|
'logo': string;
|
||||||
|
'logoContainer': string;
|
||||||
|
'logoLink': string;
|
||||||
'sidebar': string;
|
'sidebar': string;
|
||||||
|
'sidebarCloseButton': string;
|
||||||
'sidebarContainer': string;
|
'sidebarContainer': string;
|
||||||
|
'sidebarHeader': string;
|
||||||
}
|
}
|
||||||
export const cssExports: CssExports;
|
export const cssExports: CssExports;
|
||||||
export default cssExports;
|
export default cssExports;
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import classNames from 'classnames';
|
|
||||||
import React, {
|
import React, {
|
||||||
useCallback,
|
useCallback,
|
||||||
useEffect,
|
useEffect,
|
||||||
@@ -11,6 +10,8 @@ import { useDispatch } from 'react-redux';
|
|||||||
import { useLocation } from 'react-router';
|
import { useLocation } from 'react-router';
|
||||||
import QueueStatus from 'Activity/Queue/Status/QueueStatus';
|
import QueueStatus from 'Activity/Queue/Status/QueueStatus';
|
||||||
import { IconName } from 'Components/Icon';
|
import { IconName } from 'Components/Icon';
|
||||||
|
import IconButton from 'Components/Link/IconButton';
|
||||||
|
import Link from 'Components/Link/Link';
|
||||||
import OverlayScroller from 'Components/Scroller/OverlayScroller';
|
import OverlayScroller from 'Components/Scroller/OverlayScroller';
|
||||||
import Scroller from 'Components/Scroller/Scroller';
|
import Scroller from 'Components/Scroller/Scroller';
|
||||||
import usePrevious from 'Helpers/Hooks/usePrevious';
|
import usePrevious from 'Helpers/Hooks/usePrevious';
|
||||||
@@ -230,10 +231,6 @@ function PageSidebar({ isSidebarVisible, isSmallScreen }: PageSidebarProps) {
|
|||||||
transition: 'none',
|
transition: 'none',
|
||||||
transform: isSidebarVisible ? 0 : SIDEBAR_WIDTH * -1,
|
transform: isSidebarVisible ? 0 : SIDEBAR_WIDTH * -1,
|
||||||
});
|
});
|
||||||
const [sidebarStyle, setSidebarStyle] = useState({
|
|
||||||
top: dimensions.headerHeight,
|
|
||||||
height: `${window.innerHeight - HEADER_HEIGHT}px`,
|
|
||||||
});
|
|
||||||
|
|
||||||
const urlBase = window.Sonarr.urlBase;
|
const urlBase = window.Sonarr.urlBase;
|
||||||
const pathname = urlBase
|
const pathname = urlBase
|
||||||
@@ -299,22 +296,6 @@ function PageSidebar({ isSidebarVisible, isSmallScreen }: PageSidebarProps) {
|
|||||||
dispatch(setIsSidebarVisible({ isSidebarVisible: false }));
|
dispatch(setIsSidebarVisible({ isSidebarVisible: false }));
|
||||||
}, [dispatch]);
|
}, [dispatch]);
|
||||||
|
|
||||||
const handleWindowScroll = useCallback(() => {
|
|
||||||
const windowScroll =
|
|
||||||
window.scrollY == null
|
|
||||||
? document.documentElement.scrollTop
|
|
||||||
: window.scrollY;
|
|
||||||
const sidebarTop = Math.max(HEADER_HEIGHT - windowScroll, 0);
|
|
||||||
const sidebarHeight = window.innerHeight - sidebarTop;
|
|
||||||
|
|
||||||
if (isSmallScreen) {
|
|
||||||
setSidebarStyle({
|
|
||||||
top: `${sidebarTop}px`,
|
|
||||||
height: `${sidebarHeight}px`,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}, [isSmallScreen]);
|
|
||||||
|
|
||||||
const handleTouchStart = useCallback(
|
const handleTouchStart = useCallback(
|
||||||
(event: TouchEvent) => {
|
(event: TouchEvent) => {
|
||||||
const touches = event.touches;
|
const touches = event.touches;
|
||||||
@@ -396,10 +377,13 @@ function PageSidebar({ isSidebarVisible, isSmallScreen }: PageSidebarProps) {
|
|||||||
touchStartY.current = null;
|
touchStartY.current = null;
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const handleSidebarClosePress = useCallback(() => {
|
||||||
|
dispatch(setIsSidebarVisible({ isSidebarVisible: false }));
|
||||||
|
}, [dispatch]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (isSmallScreen) {
|
if (isSmallScreen) {
|
||||||
window.addEventListener('click', handleWindowClick, { capture: true });
|
window.addEventListener('click', handleWindowClick, { capture: true });
|
||||||
window.addEventListener('scroll', handleWindowScroll);
|
|
||||||
window.addEventListener('touchstart', handleTouchStart);
|
window.addEventListener('touchstart', handleTouchStart);
|
||||||
window.addEventListener('touchmove', handleTouchMove);
|
window.addEventListener('touchmove', handleTouchMove);
|
||||||
window.addEventListener('touchend', handleTouchEnd);
|
window.addEventListener('touchend', handleTouchEnd);
|
||||||
@@ -408,7 +392,6 @@ function PageSidebar({ isSidebarVisible, isSmallScreen }: PageSidebarProps) {
|
|||||||
|
|
||||||
return () => {
|
return () => {
|
||||||
window.removeEventListener('click', handleWindowClick, { capture: true });
|
window.removeEventListener('click', handleWindowClick, { capture: true });
|
||||||
window.removeEventListener('scroll', handleWindowScroll);
|
|
||||||
window.removeEventListener('touchstart', handleTouchStart);
|
window.removeEventListener('touchstart', handleTouchStart);
|
||||||
window.removeEventListener('touchmove', handleTouchMove);
|
window.removeEventListener('touchmove', handleTouchMove);
|
||||||
window.removeEventListener('touchend', handleTouchEnd);
|
window.removeEventListener('touchend', handleTouchEnd);
|
||||||
@@ -417,7 +400,6 @@ function PageSidebar({ isSidebarVisible, isSmallScreen }: PageSidebarProps) {
|
|||||||
}, [
|
}, [
|
||||||
isSmallScreen,
|
isSmallScreen,
|
||||||
handleWindowClick,
|
handleWindowClick,
|
||||||
handleWindowScroll,
|
|
||||||
handleTouchStart,
|
handleTouchStart,
|
||||||
handleTouchMove,
|
handleTouchMove,
|
||||||
handleTouchEnd,
|
handleTouchEnd,
|
||||||
@@ -456,13 +438,37 @@ function PageSidebar({ isSidebarVisible, isSmallScreen }: PageSidebarProps) {
|
|||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
ref={sidebarRef}
|
ref={sidebarRef}
|
||||||
className={classNames(styles.sidebarContainer)}
|
className={styles.sidebarContainer}
|
||||||
style={containerStyle}
|
style={containerStyle}
|
||||||
>
|
>
|
||||||
|
{isSmallScreen ? (
|
||||||
|
<div className={styles.sidebarHeader}>
|
||||||
|
<div className={styles.logoContainer}>
|
||||||
|
<Link className={styles.logoLink} to="/">
|
||||||
|
<img
|
||||||
|
className={styles.logo}
|
||||||
|
src={`${window.Sonarr.urlBase}/Content/Images/logo.svg`}
|
||||||
|
alt="Sonarr Logo"
|
||||||
|
/>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<IconButton
|
||||||
|
className={styles.sidebarCloseButton}
|
||||||
|
name={icons.CLOSE}
|
||||||
|
aria-label={translate('Close')}
|
||||||
|
size={20}
|
||||||
|
onPress={handleSidebarClosePress}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
) : null}
|
||||||
|
|
||||||
<ScrollerComponent
|
<ScrollerComponent
|
||||||
className={styles.sidebar}
|
className={styles.sidebar}
|
||||||
scrollDirection="vertical"
|
scrollDirection="vertical"
|
||||||
style={sidebarStyle}
|
style={{
|
||||||
|
height: `${window.innerHeight - HEADER_HEIGHT}px`,
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
<div>
|
<div>
|
||||||
{LINKS.map((link) => {
|
{LINKS.map((link) => {
|
||||||
|
|||||||
@@ -62,6 +62,20 @@ export const defaultState = {
|
|||||||
isPopulated: false,
|
isPopulated: false,
|
||||||
error: null,
|
error: null,
|
||||||
items: []
|
items: []
|
||||||
|
},
|
||||||
|
|
||||||
|
logFiles: {
|
||||||
|
isFetching: false,
|
||||||
|
isPopulated: false,
|
||||||
|
error: null,
|
||||||
|
items: []
|
||||||
|
},
|
||||||
|
|
||||||
|
updateLogFiles: {
|
||||||
|
isFetching: false,
|
||||||
|
isPopulated: false,
|
||||||
|
error: null,
|
||||||
|
items: []
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -80,6 +94,11 @@ export const RESTORE_BACKUP = 'system/backups/restoreBackup';
|
|||||||
export const CLEAR_RESTORE_BACKUP = 'system/backups/clearRestoreBackup';
|
export const CLEAR_RESTORE_BACKUP = 'system/backups/clearRestoreBackup';
|
||||||
export const DELETE_BACKUP = 'system/backups/deleteBackup';
|
export const DELETE_BACKUP = 'system/backups/deleteBackup';
|
||||||
|
|
||||||
|
export const FETCH_UPDATES = 'system/updates/fetchUpdates';
|
||||||
|
|
||||||
|
export const FETCH_LOG_FILES = 'system/logFiles/fetchLogFiles';
|
||||||
|
export const FETCH_UPDATE_LOG_FILES = 'system/updateLogFiles/fetchUpdateLogFiles';
|
||||||
|
|
||||||
export const RESTART = 'system/restart';
|
export const RESTART = 'system/restart';
|
||||||
export const SHUTDOWN = 'system/shutdown';
|
export const SHUTDOWN = 'system/shutdown';
|
||||||
|
|
||||||
@@ -98,6 +117,11 @@ export const restoreBackup = createThunk(RESTORE_BACKUP);
|
|||||||
export const clearRestoreBackup = createAction(CLEAR_RESTORE_BACKUP);
|
export const clearRestoreBackup = createAction(CLEAR_RESTORE_BACKUP);
|
||||||
export const deleteBackup = createThunk(DELETE_BACKUP);
|
export const deleteBackup = createThunk(DELETE_BACKUP);
|
||||||
|
|
||||||
|
export const fetchUpdates = createThunk(FETCH_UPDATES);
|
||||||
|
|
||||||
|
export const fetchLogFiles = createThunk(FETCH_LOG_FILES);
|
||||||
|
export const fetchUpdateLogFiles = createThunk(FETCH_UPDATE_LOG_FILES);
|
||||||
|
|
||||||
export const restart = createThunk(RESTART);
|
export const restart = createThunk(RESTART);
|
||||||
export const shutdown = createThunk(SHUTDOWN);
|
export const shutdown = createThunk(SHUTDOWN);
|
||||||
|
|
||||||
@@ -176,6 +200,10 @@ export const actionHandlers = handleThunks({
|
|||||||
|
|
||||||
[DELETE_BACKUP]: createRemoveItemHandler(backupsSection, '/system/backup'),
|
[DELETE_BACKUP]: createRemoveItemHandler(backupsSection, '/system/backup'),
|
||||||
|
|
||||||
|
[FETCH_UPDATES]: createFetchHandler('system.updates', '/update'),
|
||||||
|
[FETCH_LOG_FILES]: createFetchHandler('system.logFiles', '/log/file'),
|
||||||
|
[FETCH_UPDATE_LOG_FILES]: createFetchHandler('system.updateLogFiles', '/log/file/update'),
|
||||||
|
|
||||||
[RESTART]: function(getState, payload, dispatch) {
|
[RESTART]: function(getState, payload, dispatch) {
|
||||||
const promise = createAjaxRequest({
|
const promise = createAjaxRequest({
|
||||||
url: '/system/restart',
|
url: '/system/restart',
|
||||||
|
|||||||
@@ -1,39 +1,46 @@
|
|||||||
import React, { useCallback } from 'react';
|
import React, { useCallback, useEffect } from 'react';
|
||||||
import { useDispatch, useSelector } from 'react-redux';
|
import { useDispatch, useSelector } from 'react-redux';
|
||||||
|
import AppState from 'App/State/AppState';
|
||||||
import * as commandNames from 'Commands/commandNames';
|
import * as commandNames from 'Commands/commandNames';
|
||||||
import { executeCommand } from 'Store/Actions/commandActions';
|
import { executeCommand } from 'Store/Actions/commandActions';
|
||||||
|
import { fetchLogFiles } from 'Store/Actions/systemActions';
|
||||||
import createCommandExecutingSelector from 'Store/Selectors/createCommandExecutingSelector';
|
import createCommandExecutingSelector from 'Store/Selectors/createCommandExecutingSelector';
|
||||||
import LogFiles from '../LogFiles';
|
import LogFiles from '../LogFiles';
|
||||||
import useLogFiles from '../useLogFiles';
|
|
||||||
|
|
||||||
function AppLogFiles() {
|
function AppLogFiles() {
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const { data = [], isFetching, refetch } = useLogFiles();
|
const { isFetching, items } = useSelector(
|
||||||
|
(state: AppState) => state.system.logFiles
|
||||||
|
);
|
||||||
|
|
||||||
const isDeleteFilesExecuting = useSelector(
|
const isDeleteFilesExecuting = useSelector(
|
||||||
createCommandExecutingSelector(commandNames.DELETE_LOG_FILES)
|
createCommandExecutingSelector(commandNames.DELETE_LOG_FILES)
|
||||||
);
|
);
|
||||||
|
|
||||||
const handleRefreshPress = useCallback(() => {
|
const handleRefreshPress = useCallback(() => {
|
||||||
refetch();
|
dispatch(fetchLogFiles());
|
||||||
}, [refetch]);
|
}, [dispatch]);
|
||||||
|
|
||||||
const handleDeleteFilesPress = useCallback(() => {
|
const handleDeleteFilesPress = useCallback(() => {
|
||||||
dispatch(
|
dispatch(
|
||||||
executeCommand({
|
executeCommand({
|
||||||
name: commandNames.DELETE_LOG_FILES,
|
name: commandNames.DELETE_LOG_FILES,
|
||||||
commandFinished: () => {
|
commandFinished: () => {
|
||||||
refetch();
|
dispatch(fetchLogFiles());
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
}, [dispatch, refetch]);
|
}, [dispatch]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
dispatch(fetchLogFiles());
|
||||||
|
}, [dispatch]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<LogFiles
|
<LogFiles
|
||||||
isDeleteFilesExecuting={isDeleteFilesExecuting}
|
isDeleteFilesExecuting={isDeleteFilesExecuting}
|
||||||
isFetching={isFetching}
|
isFetching={isFetching}
|
||||||
items={data}
|
items={items}
|
||||||
type="app"
|
type="app"
|
||||||
onRefreshPress={handleRefreshPress}
|
onRefreshPress={handleRefreshPress}
|
||||||
onDeleteFilesPress={handleDeleteFilesPress}
|
onDeleteFilesPress={handleDeleteFilesPress}
|
||||||
|
|||||||
@@ -1,39 +1,46 @@
|
|||||||
import React, { useCallback } from 'react';
|
import React, { useCallback, useEffect } from 'react';
|
||||||
import { useDispatch, useSelector } from 'react-redux';
|
import { useDispatch, useSelector } from 'react-redux';
|
||||||
|
import AppState from 'App/State/AppState';
|
||||||
import * as commandNames from 'Commands/commandNames';
|
import * as commandNames from 'Commands/commandNames';
|
||||||
import { executeCommand } from 'Store/Actions/commandActions';
|
import { executeCommand } from 'Store/Actions/commandActions';
|
||||||
|
import { fetchUpdateLogFiles } from 'Store/Actions/systemActions';
|
||||||
import createCommandExecutingSelector from 'Store/Selectors/createCommandExecutingSelector';
|
import createCommandExecutingSelector from 'Store/Selectors/createCommandExecutingSelector';
|
||||||
import LogFiles from '../LogFiles';
|
import LogFiles from '../LogFiles';
|
||||||
import { useUpdateLogFiles } from '../useLogFiles';
|
|
||||||
|
|
||||||
function UpdateLogFiles() {
|
function UpdateLogFiles() {
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const { data = [], isFetching, refetch } = useUpdateLogFiles();
|
const { isFetching, items } = useSelector(
|
||||||
|
(state: AppState) => state.system.updateLogFiles
|
||||||
|
);
|
||||||
|
|
||||||
const isDeleteFilesExecuting = useSelector(
|
const isDeleteFilesExecuting = useSelector(
|
||||||
createCommandExecutingSelector(commandNames.DELETE_UPDATE_LOG_FILES)
|
createCommandExecutingSelector(commandNames.DELETE_UPDATE_LOG_FILES)
|
||||||
);
|
);
|
||||||
|
|
||||||
const handleRefreshPress = useCallback(() => {
|
const handleRefreshPress = useCallback(() => {
|
||||||
refetch();
|
dispatch(fetchUpdateLogFiles());
|
||||||
}, [refetch]);
|
}, [dispatch]);
|
||||||
|
|
||||||
const handleDeleteFilesPress = useCallback(() => {
|
const handleDeleteFilesPress = useCallback(() => {
|
||||||
dispatch(
|
dispatch(
|
||||||
executeCommand({
|
executeCommand({
|
||||||
name: commandNames.DELETE_UPDATE_LOG_FILES,
|
name: commandNames.DELETE_UPDATE_LOG_FILES,
|
||||||
commandFinished: () => {
|
commandFinished: () => {
|
||||||
refetch();
|
dispatch(fetchUpdateLogFiles());
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
}, [dispatch, refetch]);
|
}, [dispatch]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
dispatch(fetchUpdateLogFiles());
|
||||||
|
}, [dispatch]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<LogFiles
|
<LogFiles
|
||||||
isDeleteFilesExecuting={isDeleteFilesExecuting}
|
isDeleteFilesExecuting={isDeleteFilesExecuting}
|
||||||
isFetching={isFetching}
|
isFetching={isFetching}
|
||||||
items={data}
|
items={items}
|
||||||
type="update"
|
type="update"
|
||||||
onRefreshPress={handleRefreshPress}
|
onRefreshPress={handleRefreshPress}
|
||||||
onDeleteFilesPress={handleDeleteFilesPress}
|
onDeleteFilesPress={handleDeleteFilesPress}
|
||||||
|
|||||||
@@ -1,14 +0,0 @@
|
|||||||
import useApiQuery from 'Helpers/Hooks/useApiQuery';
|
|
||||||
import LogFile from 'typings/LogFile';
|
|
||||||
|
|
||||||
export default function useLogFiles() {
|
|
||||||
return useApiQuery<LogFile[]>({
|
|
||||||
path: '/log/file',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useUpdateLogFiles() {
|
|
||||||
return useApiQuery<LogFile[]>({
|
|
||||||
path: '/log/file/update',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
@@ -15,6 +15,7 @@ import { icons, kinds } from 'Helpers/Props';
|
|||||||
import useUpdateSettings from 'Settings/General/useUpdateSettings';
|
import useUpdateSettings from 'Settings/General/useUpdateSettings';
|
||||||
import { executeCommand } from 'Store/Actions/commandActions';
|
import { executeCommand } from 'Store/Actions/commandActions';
|
||||||
import { fetchGeneralSettings } from 'Store/Actions/settingsActions';
|
import { fetchGeneralSettings } from 'Store/Actions/settingsActions';
|
||||||
|
import { fetchUpdates } from 'Store/Actions/systemActions';
|
||||||
import createCommandExecutingSelector from 'Store/Selectors/createCommandExecutingSelector';
|
import createCommandExecutingSelector from 'Store/Selectors/createCommandExecutingSelector';
|
||||||
import createSystemStatusSelector from 'Store/Selectors/createSystemStatusSelector';
|
import createSystemStatusSelector from 'Store/Selectors/createSystemStatusSelector';
|
||||||
import createUISettingsSelector from 'Store/Selectors/createUISettingsSelector';
|
import createUISettingsSelector from 'Store/Selectors/createUISettingsSelector';
|
||||||
@@ -113,6 +114,7 @@ function Updates() {
|
|||||||
}, [setIsMajorUpdateModalOpen]);
|
}, [setIsMajorUpdateModalOpen]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
dispatch(fetchUpdates());
|
||||||
dispatch(fetchGeneralSettings());
|
dispatch(fetchGeneralSettings());
|
||||||
}, [dispatch]);
|
}, [dispatch]);
|
||||||
|
|
||||||
|
|||||||
@@ -1,41 +0,0 @@
|
|||||||
using NzbDrone.Common.Disk;
|
|
||||||
using NzbDrone.Common.EnvironmentInfo;
|
|
||||||
using NzbDrone.Common.Extensions;
|
|
||||||
using NzbDrone.Core.Configuration;
|
|
||||||
using Sonarr.Http;
|
|
||||||
|
|
||||||
namespace Sonarr.Api.V5.Logs;
|
|
||||||
|
|
||||||
[V5ApiController("log/file")]
|
|
||||||
public class LogFileController : LogFileControllerBase
|
|
||||||
{
|
|
||||||
private readonly IAppFolderInfo _appFolderInfo;
|
|
||||||
private readonly IDiskProvider _diskProvider;
|
|
||||||
|
|
||||||
public LogFileController(IAppFolderInfo appFolderInfo,
|
|
||||||
IDiskProvider diskProvider,
|
|
||||||
IConfigFileProvider configFileProvider)
|
|
||||||
: base(diskProvider, configFileProvider, "")
|
|
||||||
{
|
|
||||||
_appFolderInfo = appFolderInfo;
|
|
||||||
_diskProvider = diskProvider;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override IEnumerable<string> GetLogFiles()
|
|
||||||
{
|
|
||||||
return _diskProvider.GetFiles(_appFolderInfo.GetLogFolder(), false);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override string GetLogFilePath(string filename)
|
|
||||||
{
|
|
||||||
return Path.Combine(_appFolderInfo.GetLogFolder(), filename);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override string DownloadUrlRoot
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
return "logfile";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,71 +0,0 @@
|
|||||||
using Microsoft.AspNetCore.Mvc;
|
|
||||||
using NLog;
|
|
||||||
using NzbDrone.Common.Disk;
|
|
||||||
using NzbDrone.Core.Configuration;
|
|
||||||
|
|
||||||
namespace Sonarr.Api.V5.Logs;
|
|
||||||
|
|
||||||
public abstract class LogFileControllerBase : Controller
|
|
||||||
{
|
|
||||||
protected const string LOGFILE_ROUTE = @"/(?<filename>[-.a-zA-Z0-9]+?\.txt)";
|
|
||||||
protected string _resource;
|
|
||||||
|
|
||||||
private readonly IDiskProvider _diskProvider;
|
|
||||||
private readonly IConfigFileProvider _configFileProvider;
|
|
||||||
|
|
||||||
public LogFileControllerBase(IDiskProvider diskProvider,
|
|
||||||
IConfigFileProvider configFileProvider,
|
|
||||||
string resource)
|
|
||||||
{
|
|
||||||
_diskProvider = diskProvider;
|
|
||||||
_configFileProvider = configFileProvider;
|
|
||||||
_resource = resource;
|
|
||||||
}
|
|
||||||
|
|
||||||
[HttpGet]
|
|
||||||
[Produces("application/json")]
|
|
||||||
public List<LogFileResource> GetLogFilesResponse()
|
|
||||||
{
|
|
||||||
var result = new List<LogFileResource>();
|
|
||||||
|
|
||||||
var files = GetLogFiles().ToList();
|
|
||||||
|
|
||||||
for (var i = 0; i < files.Count; i++)
|
|
||||||
{
|
|
||||||
var file = files[i];
|
|
||||||
var filename = Path.GetFileName(file);
|
|
||||||
|
|
||||||
result.Add(new LogFileResource
|
|
||||||
{
|
|
||||||
Id = i + 1,
|
|
||||||
Filename = filename,
|
|
||||||
LastWriteTime = _diskProvider.FileGetLastWrite(file),
|
|
||||||
ContentsUrl = string.Format("{0}/api/v1/{1}/{2}", _configFileProvider.UrlBase, _resource, filename),
|
|
||||||
DownloadUrl = string.Format("{0}/{1}/{2}", _configFileProvider.UrlBase, DownloadUrlRoot, filename)
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return result.OrderByDescending(l => l.LastWriteTime).ToList();
|
|
||||||
}
|
|
||||||
|
|
||||||
[HttpGet(@"{filename:regex([[-.a-zA-Z0-9]]+?\.txt)}")]
|
|
||||||
[Produces("text/plain")]
|
|
||||||
public IActionResult GetLogFileResponse(string filename)
|
|
||||||
{
|
|
||||||
LogManager.Flush();
|
|
||||||
|
|
||||||
var filePath = GetLogFilePath(filename);
|
|
||||||
|
|
||||||
if (!_diskProvider.FileExists(filePath))
|
|
||||||
{
|
|
||||||
return NotFound();
|
|
||||||
}
|
|
||||||
|
|
||||||
return PhysicalFile(filePath, "text/plain");
|
|
||||||
}
|
|
||||||
|
|
||||||
protected abstract IEnumerable<string> GetLogFiles();
|
|
||||||
protected abstract string GetLogFilePath(string filename);
|
|
||||||
|
|
||||||
protected abstract string DownloadUrlRoot { get; }
|
|
||||||
}
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
using Sonarr.Http.REST;
|
|
||||||
|
|
||||||
namespace Sonarr.Api.V5.Logs;
|
|
||||||
|
|
||||||
public class LogFileResource : RestResource
|
|
||||||
{
|
|
||||||
public required string Filename { get; set; }
|
|
||||||
public required DateTime LastWriteTime { get; set; }
|
|
||||||
public required string ContentsUrl { get; set; }
|
|
||||||
public required string DownloadUrl { get; set; }
|
|
||||||
}
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
using System.Text.RegularExpressions;
|
|
||||||
using NzbDrone.Common.Disk;
|
|
||||||
using NzbDrone.Common.EnvironmentInfo;
|
|
||||||
using NzbDrone.Common.Extensions;
|
|
||||||
using NzbDrone.Core.Configuration;
|
|
||||||
using Sonarr.Http;
|
|
||||||
|
|
||||||
namespace Sonarr.Api.V5.Logs;
|
|
||||||
|
|
||||||
[V5ApiController("log/file/update")]
|
|
||||||
public class UpdateLogFileController : LogFileControllerBase
|
|
||||||
{
|
|
||||||
private readonly IAppFolderInfo _appFolderInfo;
|
|
||||||
private readonly IDiskProvider _diskProvider;
|
|
||||||
|
|
||||||
public UpdateLogFileController(IAppFolderInfo appFolderInfo,
|
|
||||||
IDiskProvider diskProvider,
|
|
||||||
IConfigFileProvider configFileProvider)
|
|
||||||
: base(diskProvider, configFileProvider, "update")
|
|
||||||
{
|
|
||||||
_appFolderInfo = appFolderInfo;
|
|
||||||
_diskProvider = diskProvider;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override IEnumerable<string> GetLogFiles()
|
|
||||||
{
|
|
||||||
if (!_diskProvider.FolderExists(_appFolderInfo.GetUpdateLogFolder()))
|
|
||||||
{
|
|
||||||
return Enumerable.Empty<string>();
|
|
||||||
}
|
|
||||||
|
|
||||||
return _diskProvider.GetFiles(_appFolderInfo.GetUpdateLogFolder(), false)
|
|
||||||
.Where(f => Regex.IsMatch(Path.GetFileName(f), LOGFILE_ROUTE.TrimStart('/'), RegexOptions.IgnoreCase))
|
|
||||||
.ToList();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override string GetLogFilePath(string filename)
|
|
||||||
{
|
|
||||||
return Path.Combine(_appFolderInfo.GetUpdateLogFolder(), filename);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override string DownloadUrlRoot
|
|
||||||
{
|
|
||||||
get
|
|
||||||
{
|
|
||||||
return "updatelogfile";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user