@knpkv/jira-clockify 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +121 -0
- package/dist/package.json +47 -0
- package/dist/src/bin.js +28 -0
- package/dist/src/bin.js.map +1 -0
- package/dist/src/cli/auth.js +169 -0
- package/dist/src/cli/auth.js.map +1 -0
- package/dist/src/cli/config.js +107 -0
- package/dist/src/cli/config.js.map +1 -0
- package/dist/src/cli/fuzzySelect.js +149 -0
- package/dist/src/cli/fuzzySelect.js.map +1 -0
- package/dist/src/cli/layers.js +59 -0
- package/dist/src/cli/layers.js.map +1 -0
- package/dist/src/cli/list.js +27 -0
- package/dist/src/cli/list.js.map +1 -0
- package/dist/src/cli/setup.js +134 -0
- package/dist/src/cli/setup.js.map +1 -0
- package/dist/src/cli/timer/discard.js +33 -0
- package/dist/src/cli/timer/discard.js.map +1 -0
- package/dist/src/cli/timer/edit.js +139 -0
- package/dist/src/cli/timer/edit.js.map +1 -0
- package/dist/src/cli/timer/index.js +12 -0
- package/dist/src/cli/timer/index.js.map +1 -0
- package/dist/src/cli/timer/log.js +96 -0
- package/dist/src/cli/timer/log.js.map +1 -0
- package/dist/src/cli/timer/start.js +156 -0
- package/dist/src/cli/timer/start.js.map +1 -0
- package/dist/src/cli/timer/status.js +80 -0
- package/dist/src/cli/timer/status.js.map +1 -0
- package/dist/src/cli/timer/stop.js +96 -0
- package/dist/src/cli/timer/stop.js.map +1 -0
- package/dist/src/cli/timer.js +7 -0
- package/dist/src/cli/timer.js.map +1 -0
- package/dist/src/main.js +24 -0
- package/dist/src/main.js.map +1 -0
- package/dist/src/services/ClockifyAuth.js +77 -0
- package/dist/src/services/ClockifyAuth.js.map +1 -0
- package/dist/src/services/ConfigService.js +66 -0
- package/dist/src/services/ConfigService.js.map +1 -0
- package/dist/src/services/StateWriter.js +69 -0
- package/dist/src/services/StateWriter.js.map +1 -0
- package/dist/src/services/TicketService.js +106 -0
- package/dist/src/services/TicketService.js.map +1 -0
- package/dist/src/services/TimerService.js +271 -0
- package/dist/src/services/TimerService.js.map +1 -0
- package/dist/src/tui/App.js +204 -0
- package/dist/src/tui/App.js.map +1 -0
- package/dist/src/tui/atoms/runtime.js +9 -0
- package/dist/src/tui/atoms/runtime.js.map +1 -0
- package/dist/src/tui/atoms/tickets.js +17 -0
- package/dist/src/tui/atoms/tickets.js.map +1 -0
- package/dist/src/tui/atoms/timer.js +31 -0
- package/dist/src/tui/atoms/timer.js.map +1 -0
- package/dist/src/tui/atoms/ui.js +10 -0
- package/dist/src/tui/atoms/ui.js.map +1 -0
- package/dist/src/tui/components/BigTimer.js +60 -0
- package/dist/src/tui/components/BigTimer.js.map +1 -0
- package/dist/src/tui/components/Footer.js +16 -0
- package/dist/src/tui/components/Footer.js.map +1 -0
- package/dist/src/tui/components/Header.js +16 -0
- package/dist/src/tui/components/Header.js.map +1 -0
- package/dist/src/tui/components/PopupInput.js +30 -0
- package/dist/src/tui/components/PopupInput.js.map +1 -0
- package/dist/src/tui/components/PopupMessage.js +47 -0
- package/dist/src/tui/components/PopupMessage.js.map +1 -0
- package/dist/src/tui/components/TicketList.js +40 -0
- package/dist/src/tui/components/TicketList.js.map +1 -0
- package/dist/src/tui/components/TicketRow.js +48 -0
- package/dist/src/tui/components/TicketRow.js.map +1 -0
- package/dist/src/tui/components/TimerDisplay.js +30 -0
- package/dist/src/tui/components/TimerDisplay.js.map +1 -0
- package/dist/src/tui/components/index.js +12 -0
- package/dist/src/tui/components/index.js.map +1 -0
- package/dist/src/tui/context/theme.js +15 -0
- package/dist/src/tui/context/theme.js.map +1 -0
- package/dist/src/tui/hooks/useElapsedTimer.js +35 -0
- package/dist/src/tui/hooks/useElapsedTimer.js.map +1 -0
- package/dist/src/tui/hooks/useTerminalSize.js +32 -0
- package/dist/src/tui/hooks/useTerminalSize.js.map +1 -0
- package/dist/src/utils/time.js +23 -0
- package/dist/src/utils/time.js.map +1 -0
- package/dist/test/TimerService.test.js +355 -0
- package/dist/test/TimerService.test.js.map +1 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/nvim/lua/jcf/branch.lua +26 -0
- package/nvim/lua/jcf/float.lua +87 -0
- package/nvim/lua/jcf/init.lua +70 -0
- package/nvim/lua/jcf/state.lua +56 -0
- package/nvim/lua/jcf/statusline.lua +31 -0
- package/nvim/lua/jcf/telescope.lua +50 -0
- package/nvim/plugin/jcf.vim +2 -0
- package/package.json +47 -0
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "@opentui/react/jsx-runtime";
|
|
2
|
+
/**
|
|
3
|
+
* Compact timer widget showing elapsed time, ticket key, and project.
|
|
4
|
+
*
|
|
5
|
+
* @internal
|
|
6
|
+
*/
|
|
7
|
+
import { formatDuration, formatElapsed } from "../../utils/time.js";
|
|
8
|
+
import { useElapsedTimer } from "../hooks/useElapsedTimer.js";
|
|
9
|
+
export function TimerDisplay() {
|
|
10
|
+
const { elapsed, timerState } = useElapsedTimer();
|
|
11
|
+
if (!timerState?.active) {
|
|
12
|
+
return (_jsx("box", { style: {
|
|
13
|
+
height: 3,
|
|
14
|
+
width: "100%",
|
|
15
|
+
flexDirection: "column",
|
|
16
|
+
alignItems: "center",
|
|
17
|
+
justifyContent: "center",
|
|
18
|
+
backgroundColor: "#0f0f1a"
|
|
19
|
+
}, children: _jsx("text", { fg: "#555555", children: "No active timer \u2014 press s or Enter on a ticket to start" }) }));
|
|
20
|
+
}
|
|
21
|
+
return (_jsxs("box", { style: {
|
|
22
|
+
height: 5,
|
|
23
|
+
width: "100%",
|
|
24
|
+
flexDirection: "column",
|
|
25
|
+
backgroundColor: "#0a1628",
|
|
26
|
+
paddingLeft: 2,
|
|
27
|
+
paddingTop: 1
|
|
28
|
+
}, children: [_jsxs("box", { style: { flexDirection: "row", gap: 2 }, children: [_jsx("text", { fg: "#00CC66", style: { fontWeight: "bold" }, children: "\u25CF" }), _jsx("text", { fg: "#00CCFF", style: { fontWeight: "bold" }, children: formatElapsed(elapsed) }), _jsx("text", { fg: "#888888", children: formatDuration(elapsed) })] }), _jsxs("box", { style: { flexDirection: "row", gap: 1, paddingLeft: 2 }, children: [_jsx("text", { fg: "#FFFFFF", style: { fontWeight: "bold" }, children: timerState.ticketKey }), _jsx("text", { fg: "#AAAAAA", children: timerState.summary ?? "" })] }), _jsxs("box", { style: { flexDirection: "row", gap: 2, paddingLeft: 2 }, children: [timerState.projectId ? _jsx("text", { fg: "#555555", children: "proj" }) : null, timerState.billable !== null ? (_jsx("text", { fg: "#555555", children: timerState.billable ? "billable" : "non-billable" })) : null] })] }));
|
|
29
|
+
}
|
|
30
|
+
//# sourceMappingURL=TimerDisplay.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"TimerDisplay.js","sourceRoot":"","sources":["../../../../src/tui/components/TimerDisplay.tsx"],"names":[],"mappings":";AAAA;;;;GAIG;AACH,OAAO,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAA;AACnE,OAAO,EAAE,eAAe,EAAE,MAAM,6BAA6B,CAAA;AAE7D,MAAM,UAAU,YAAY;IAC1B,MAAM,EAAE,OAAO,EAAE,UAAU,EAAE,GAAG,eAAe,EAAE,CAAA;IAEjD,IAAI,CAAC,UAAU,EAAE,MAAM,EAAE,CAAC;QACxB,OAAO,CACL,cACE,KAAK,EACH;gBACE,MAAM,EAAE,CAAC;gBACT,KAAK,EAAE,MAAM;gBACb,aAAa,EAAE,QAAQ;gBACvB,UAAU,EAAE,QAAQ;gBACpB,cAAc,EAAE,QAAQ;gBACxB,eAAe,EAAE,SAAS;aACpB,YAGV,eAAM,EAAE,EAAC,SAAS,6EAA+D,GAC7E,CACP,CAAA;IACH,CAAC;IAED,OAAO,CACL,eACE,KAAK,EACH;YACE,MAAM,EAAE,CAAC;YACT,KAAK,EAAE,MAAM;YACb,aAAa,EAAE,QAAQ;YACvB,eAAe,EAAE,SAAS;YAC1B,WAAW,EAAE,CAAC;YACd,UAAU,EAAE,CAAC;SACP,aAGV,eAAK,KAAK,EAAE,EAAE,aAAa,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,EAAE,aAC1C,eAAM,EAAE,EAAC,SAAS,EAAC,KAAK,EAAE,EAAE,UAAU,EAAE,MAAM,EAAS,uBAEhD,EACP,eAAM,EAAE,EAAC,SAAS,EAAC,KAAK,EAAE,EAAE,UAAU,EAAE,MAAM,EAAS,YACpD,aAAa,CAAC,OAAO,CAAC,GAClB,EACP,eAAM,EAAE,EAAC,SAAS,YAAE,cAAc,CAAC,OAAO,CAAC,GAAQ,IAC/C,EACN,eAAK,KAAK,EAAE,EAAE,aAAa,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,EAAE,WAAW,EAAE,CAAC,EAAS,aACjE,eAAM,EAAE,EAAC,SAAS,EAAC,KAAK,EAAE,EAAE,UAAU,EAAE,MAAM,EAAS,YACpD,UAAU,CAAC,SAAS,GAChB,EACP,eAAM,EAAE,EAAC,SAAS,YAAE,UAAU,CAAC,OAAO,IAAI,EAAE,GAAQ,IAChD,EACN,eAAK,KAAK,EAAE,EAAE,aAAa,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,EAAE,WAAW,EAAE,CAAC,EAAS,aAChE,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC,eAAM,EAAE,EAAC,SAAS,qBAAY,CAAC,CAAC,CAAC,IAAI,EAC5D,UAAU,CAAC,QAAQ,KAAK,IAAI,CAAC,CAAC,CAAC,CAC9B,eAAM,EAAE,EAAC,SAAS,YAAE,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,cAAc,GAAQ,CAC9E,CAAC,CAAC,CAAC,IAAI,IACJ,IACF,CACP,CAAA;AACH,CAAC"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Barrel export for TUI components.
|
|
3
|
+
*
|
|
4
|
+
* @module
|
|
5
|
+
*/
|
|
6
|
+
export { BigTimer } from "./BigTimer.js";
|
|
7
|
+
export { Footer } from "./Footer.js";
|
|
8
|
+
export { Header } from "./Header.js";
|
|
9
|
+
export { TicketList } from "./TicketList.js";
|
|
10
|
+
export { TicketRow } from "./TicketRow.js";
|
|
11
|
+
export { TimerDisplay } from "./TimerDisplay.js";
|
|
12
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/tui/components/index.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AACH,OAAO,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAA;AACxC,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAA;AACpC,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAA;AACpC,OAAO,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAA;AAC5C,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAA;AAC1C,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAA"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { jsx as _jsx } from "@opentui/react/jsx-runtime";
|
|
2
|
+
/**
|
|
3
|
+
* React context provider for TUI color theme.
|
|
4
|
+
*
|
|
5
|
+
* @internal
|
|
6
|
+
*/
|
|
7
|
+
import { createContext, useContext } from "react";
|
|
8
|
+
const ThemeContext = createContext({ theme: "dark" });
|
|
9
|
+
export function ThemeProvider({ children }) {
|
|
10
|
+
return _jsx(ThemeContext.Provider, { value: { theme: "dark" }, children: children });
|
|
11
|
+
}
|
|
12
|
+
export function useTheme() {
|
|
13
|
+
return useContext(ThemeContext);
|
|
14
|
+
}
|
|
15
|
+
//# sourceMappingURL=theme.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"theme.js","sourceRoot":"","sources":["../../../../src/tui/context/theme.tsx"],"names":[],"mappings":";AAAA;;;;GAIG;AACH,OAAO,EAAE,aAAa,EAAE,UAAU,EAAE,MAAM,OAAO,CAAA;AAMjD,MAAM,YAAY,GAAG,aAAa,CAAoB,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAA;AAExE,MAAM,UAAU,aAAa,CAAC,EAAE,QAAQ,EAAoD;IAC1F,OAAO,KAAC,YAAY,CAAC,QAAQ,IAAC,KAAK,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,YAAG,QAAQ,GAAyB,CAAA;AAC5F,CAAC;AAED,MAAM,UAAU,QAAQ;IACtB,OAAO,UAAU,CAAC,YAAY,CAAC,CAAA;AACjC,CAAC"}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Hook that ticks elapsed seconds while a timer is active.
|
|
3
|
+
*
|
|
4
|
+
* @internal
|
|
5
|
+
*/
|
|
6
|
+
import { Result, useAtomSet, useAtomValue } from "@effect-atom/atom-react";
|
|
7
|
+
import { useEffect, useRef } from "react";
|
|
8
|
+
import { elapsedAtom, timerStateAtom } from "../atoms/timer.js";
|
|
9
|
+
export function useElapsedTimer() {
|
|
10
|
+
const timerResult = useAtomValue(timerStateAtom);
|
|
11
|
+
const elapsed = useAtomValue(elapsedAtom);
|
|
12
|
+
const setElapsed = useAtomSet(elapsedAtom);
|
|
13
|
+
const intervalRef = useRef(null);
|
|
14
|
+
const timerState = Result.isSuccess(timerResult) ? timerResult.value : null;
|
|
15
|
+
useEffect(() => {
|
|
16
|
+
if (timerState?.active && timerState.startedAt) {
|
|
17
|
+
const startTime = timerState.startedAt.getTime();
|
|
18
|
+
const tick = () => setElapsed(Math.floor((Date.now() - startTime) / 1000));
|
|
19
|
+
tick();
|
|
20
|
+
intervalRef.current = setInterval(tick, 1000);
|
|
21
|
+
return () => {
|
|
22
|
+
if (intervalRef.current)
|
|
23
|
+
clearInterval(intervalRef.current);
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
else {
|
|
27
|
+
setElapsed(0);
|
|
28
|
+
if (intervalRef.current)
|
|
29
|
+
clearInterval(intervalRef.current);
|
|
30
|
+
}
|
|
31
|
+
return undefined;
|
|
32
|
+
}, [timerState?.active, timerState?.startedAt, setElapsed]);
|
|
33
|
+
return { timerState, elapsed };
|
|
34
|
+
}
|
|
35
|
+
//# sourceMappingURL=useElapsedTimer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"useElapsedTimer.js","sourceRoot":"","sources":["../../../../src/tui/hooks/useElapsedTimer.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AACH,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,yBAAyB,CAAA;AAC1E,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM,OAAO,CAAA;AAEzC,OAAO,EAAE,WAAW,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAA;AAE/D,MAAM,UAAU,eAAe;IAC7B,MAAM,WAAW,GAAG,YAAY,CAAC,cAAc,CAAC,CAAA;IAChD,MAAM,OAAO,GAAG,YAAY,CAAC,WAAW,CAAC,CAAA;IACzC,MAAM,UAAU,GAAG,UAAU,CAAC,WAAW,CAAC,CAAA;IAC1C,MAAM,WAAW,GAAG,MAAM,CAAwC,IAAI,CAAC,CAAA;IAEvE,MAAM,UAAU,GAAsB,MAAM,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAA;IAE9F,SAAS,CAAC,GAAG,EAAE;QACb,IAAI,UAAU,EAAE,MAAM,IAAI,UAAU,CAAC,SAAS,EAAE,CAAC;YAC/C,MAAM,SAAS,GAAG,UAAU,CAAC,SAAS,CAAC,OAAO,EAAE,CAAA;YAChD,MAAM,IAAI,GAAG,GAAG,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC,GAAG,IAAI,CAAC,CAAC,CAAA;YAC1E,IAAI,EAAE,CAAA;YACN,WAAW,CAAC,OAAO,GAAG,WAAW,CAAC,IAAI,EAAE,IAAI,CAAC,CAAA;YAC7C,OAAO,GAAG,EAAE;gBACV,IAAI,WAAW,CAAC,OAAO;oBAAE,aAAa,CAAC,WAAW,CAAC,OAAO,CAAC,CAAA;YAC7D,CAAC,CAAA;QACH,CAAC;aAAM,CAAC;YACN,UAAU,CAAC,CAAC,CAAC,CAAA;YACb,IAAI,WAAW,CAAC,OAAO;gBAAE,aAAa,CAAC,WAAW,CAAC,OAAO,CAAC,CAAA;QAC7D,CAAC;QACD,OAAO,SAAS,CAAA;IAClB,CAAC,EAAE,CAAC,UAAU,EAAE,MAAM,EAAE,UAAU,EAAE,SAAS,EAAE,UAAU,CAAC,CAAC,CAAA;IAE3D,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,CAAA;AAChC,CAAC"}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Hook that tracks terminal column width via Node stdout.
|
|
3
|
+
*
|
|
4
|
+
* Uses `node:process` import instead of the global — keeps the dependency explicit
|
|
5
|
+
* and avoids bare `process` globals. This is a TUI boundary: React hooks require
|
|
6
|
+
* synchronous access to terminal dimensions, which Effect's Terminal service can't
|
|
7
|
+
* provide without breaking the React render contract.
|
|
8
|
+
*
|
|
9
|
+
* @internal
|
|
10
|
+
*/
|
|
11
|
+
import nodeProcess from "node:process";
|
|
12
|
+
import { useEffect, useState } from "react";
|
|
13
|
+
export function useTerminalSize() {
|
|
14
|
+
const [cols, setCols] = useState(nodeProcess.stdout.columns ?? 80);
|
|
15
|
+
useEffect(() => {
|
|
16
|
+
const handler = () => setCols(nodeProcess.stdout.columns ?? 80);
|
|
17
|
+
nodeProcess.stdout.on("resize", handler);
|
|
18
|
+
return () => {
|
|
19
|
+
nodeProcess.stdout.off("resize", handler);
|
|
20
|
+
};
|
|
21
|
+
}, []);
|
|
22
|
+
return cols;
|
|
23
|
+
}
|
|
24
|
+
export function useDisplayMode() {
|
|
25
|
+
const cols = useTerminalSize();
|
|
26
|
+
if (cols < 40)
|
|
27
|
+
return "minimal";
|
|
28
|
+
if (cols < 80)
|
|
29
|
+
return "compact";
|
|
30
|
+
return "full";
|
|
31
|
+
}
|
|
32
|
+
//# sourceMappingURL=useTerminalSize.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"useTerminalSize.js","sourceRoot":"","sources":["../../../../src/tui/hooks/useTerminalSize.tsx"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AACH,OAAO,WAAW,MAAM,cAAc,CAAA;AACtC,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAA;AAG3C,MAAM,UAAU,eAAe;IAC7B,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,QAAQ,CAAC,WAAW,CAAC,MAAM,CAAC,OAAO,IAAI,EAAE,CAAC,CAAA;IAElE,SAAS,CAAC,GAAG,EAAE;QACb,MAAM,OAAO,GAAG,GAAG,EAAE,CAAC,OAAO,CAAC,WAAW,CAAC,MAAM,CAAC,OAAO,IAAI,EAAE,CAAC,CAAA;QAC/D,WAAW,CAAC,MAAM,CAAC,EAAE,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAA;QACxC,OAAO,GAAG,EAAE;YACV,WAAW,CAAC,MAAM,CAAC,GAAG,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAA;QAC3C,CAAC,CAAA;IACH,CAAC,EAAE,EAAE,CAAC,CAAA;IAEN,OAAO,IAAI,CAAA;AACb,CAAC;AAED,MAAM,UAAU,cAAc;IAC5B,MAAM,IAAI,GAAG,eAAe,EAAE,CAAA;IAC9B,IAAI,IAAI,GAAG,EAAE;QAAE,OAAO,SAAS,CAAA;IAC/B,IAAI,IAAI,GAAG,EAAE;QAAE,OAAO,SAAS,CAAA;IAC/B,OAAO,MAAM,CAAA;AACf,CAAC"}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared time formatting utilities.
|
|
3
|
+
*
|
|
4
|
+
* @module
|
|
5
|
+
*/
|
|
6
|
+
/** Format seconds as `HH:MM:SS`. */
|
|
7
|
+
export function formatElapsed(seconds) {
|
|
8
|
+
const h = Math.floor(seconds / 3600);
|
|
9
|
+
const m = Math.floor((seconds % 3600) / 60);
|
|
10
|
+
const s = seconds % 60;
|
|
11
|
+
return `${String(h).padStart(2, "0")}:${String(m).padStart(2, "0")}:${String(s).padStart(2, "0")}`;
|
|
12
|
+
}
|
|
13
|
+
/** Format seconds as human-readable duration (`1h 23m` or `45s`). */
|
|
14
|
+
export function formatDuration(seconds) {
|
|
15
|
+
if (seconds < 60)
|
|
16
|
+
return `${seconds}s`;
|
|
17
|
+
if (seconds < 3600)
|
|
18
|
+
return `${Math.floor(seconds / 60)}m ${seconds % 60}s`;
|
|
19
|
+
const h = Math.floor(seconds / 3600);
|
|
20
|
+
const m = Math.floor((seconds % 3600) / 60);
|
|
21
|
+
return `${h}h ${m}m`;
|
|
22
|
+
}
|
|
23
|
+
//# sourceMappingURL=time.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"time.js","sourceRoot":"","sources":["../../../src/utils/time.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,oCAAoC;AACpC,MAAM,UAAU,aAAa,CAAC,OAAe;IAC3C,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,CAAA;IACpC,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC,CAAA;IAC3C,MAAM,CAAC,GAAG,OAAO,GAAG,EAAE,CAAA;IACtB,OAAO,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAA;AACpG,CAAC;AAED,qEAAqE;AACrE,MAAM,UAAU,cAAc,CAAC,OAAe;IAC5C,IAAI,OAAO,GAAG,EAAE;QAAE,OAAO,GAAG,OAAO,GAAG,CAAA;IACtC,IAAI,OAAO,GAAG,IAAI;QAAE,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,EAAE,CAAC,KAAK,OAAO,GAAG,EAAE,GAAG,CAAA;IAC1E,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,CAAA;IACpC,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC,CAAA;IAC3C,OAAO,GAAG,CAAC,KAAK,CAAC,GAAG,CAAA;AACtB,CAAC"}
|
|
@@ -0,0 +1,355 @@
|
|
|
1
|
+
import * as HttpClient from "@effect/platform/HttpClient";
|
|
2
|
+
import * as HttpClientResponse from "@effect/platform/HttpClientResponse";
|
|
3
|
+
import { describe, expect, it } from "@effect/vitest";
|
|
4
|
+
import { ClockifyApiClient } from "@knpkv/clockify-api-client";
|
|
5
|
+
import { JiraApiClient } from "@knpkv/jira-api-client";
|
|
6
|
+
import { JiraAuth } from "@knpkv/jira-cli/JiraAuth";
|
|
7
|
+
import * as Effect from "effect/Effect";
|
|
8
|
+
import * as Layer from "effect/Layer";
|
|
9
|
+
import * as Redacted from "effect/Redacted";
|
|
10
|
+
import * as SubscriptionRef from "effect/SubscriptionRef";
|
|
11
|
+
import { ClockifyAuth } from "../src/services/ClockifyAuth.js";
|
|
12
|
+
import { ConfigService } from "../src/services/ConfigService.js";
|
|
13
|
+
import { StateWriter } from "../src/services/StateWriter.js";
|
|
14
|
+
import { layer as timerLayer, TimerError, TimerService } from "../src/services/TimerService.js";
|
|
15
|
+
// ---------------------------------------------------------------------------
|
|
16
|
+
// Helpers
|
|
17
|
+
// ---------------------------------------------------------------------------
|
|
18
|
+
const WORKSPACE_ID = "ws-1";
|
|
19
|
+
const USER_ID = "user-1";
|
|
20
|
+
const makeTicket = (overrides) => ({
|
|
21
|
+
key: "PROJ-123",
|
|
22
|
+
summary: "Fix the widget",
|
|
23
|
+
status: "In Progress",
|
|
24
|
+
priority: "High",
|
|
25
|
+
assignee: "dev",
|
|
26
|
+
type: "Bug",
|
|
27
|
+
labels: ["backend"],
|
|
28
|
+
updated: new Date().toISOString(),
|
|
29
|
+
...overrides
|
|
30
|
+
});
|
|
31
|
+
let createdEntries = [];
|
|
32
|
+
let updatedEntries = [];
|
|
33
|
+
let deletedEntries = [];
|
|
34
|
+
let stoppedTimers = [];
|
|
35
|
+
const resetCaptures = () => {
|
|
36
|
+
createdEntries = [];
|
|
37
|
+
updatedEntries = [];
|
|
38
|
+
deletedEntries = [];
|
|
39
|
+
stoppedTimers = [];
|
|
40
|
+
};
|
|
41
|
+
const makeTimeEntry = (id, description, startedAt, projectId) => ({
|
|
42
|
+
id,
|
|
43
|
+
description,
|
|
44
|
+
billable: true,
|
|
45
|
+
...(projectId ? { projectId } : {}),
|
|
46
|
+
userId: USER_ID,
|
|
47
|
+
workspaceId: WORKSPACE_ID,
|
|
48
|
+
timeInterval: { start: startedAt.toISOString() },
|
|
49
|
+
tagIds: [],
|
|
50
|
+
type: "REGULAR",
|
|
51
|
+
isLocked: false
|
|
52
|
+
});
|
|
53
|
+
const mockClockify = {
|
|
54
|
+
api: {},
|
|
55
|
+
getUser: () => Effect.succeed({
|
|
56
|
+
id: USER_ID,
|
|
57
|
+
name: "Test",
|
|
58
|
+
email: "t@t.com",
|
|
59
|
+
activeWorkspace: WORKSPACE_ID,
|
|
60
|
+
defaultWorkspace: WORKSPACE_ID,
|
|
61
|
+
profilePicture: "",
|
|
62
|
+
status: "ACTIVE"
|
|
63
|
+
}),
|
|
64
|
+
getWorkspaces: () => Effect.succeed([{ id: WORKSPACE_ID, name: "WS", imageUrl: "" }]),
|
|
65
|
+
getProjects: () => Effect.succeed([]),
|
|
66
|
+
getProjectByName: () => Effect.succeed(null),
|
|
67
|
+
createTimeEntry: (workspaceId, params) => {
|
|
68
|
+
createdEntries.push({ workspaceId, params });
|
|
69
|
+
return Effect.succeed(makeTimeEntry("entry-1", params.description, new Date(params.start)));
|
|
70
|
+
},
|
|
71
|
+
stopTimer: (workspaceId, userId, params) => {
|
|
72
|
+
stoppedTimers.push({ workspaceId, userId, params });
|
|
73
|
+
return Effect.succeed(makeTimeEntry("entry-1", "", new Date()));
|
|
74
|
+
},
|
|
75
|
+
getTimeEntries: () => Effect.succeed([]),
|
|
76
|
+
getRunningTimer: () => Effect.succeed(null),
|
|
77
|
+
getTags: () => Effect.succeed([]),
|
|
78
|
+
createTag: (_ws, name) => Effect.succeed({ id: `tag-${name}`, name, workspaceId: WORKSPACE_ID, archived: false }),
|
|
79
|
+
findOrCreateTag: (_ws, name) => Effect.succeed({ id: `tag-${name}`, name, workspaceId: WORKSPACE_ID, archived: false }),
|
|
80
|
+
getTimeEntry: (_ws, id) => Effect.succeed(makeTimeEntry(id, "[PROJ-123] Fix the widget", new Date())),
|
|
81
|
+
deleteTimeEntry: (workspaceId, id) => {
|
|
82
|
+
deletedEntries.push({ workspaceId, id });
|
|
83
|
+
return Effect.void;
|
|
84
|
+
},
|
|
85
|
+
updateTimeEntry: (workspaceId, id, params) => {
|
|
86
|
+
updatedEntries.push({ workspaceId, id, params });
|
|
87
|
+
return Effect.succeed(makeTimeEntry(id, "", new Date()));
|
|
88
|
+
}
|
|
89
|
+
};
|
|
90
|
+
const MockClockifyLayer = Layer.succeed(ClockifyApiClient, mockClockify);
|
|
91
|
+
// JiraApiClient — unused by start/stop logic directly, but required by layer construction
|
|
92
|
+
const MockJiraApiClientLayer = Layer.succeed(JiraApiClient, { v3: {} });
|
|
93
|
+
const MockClockifyAuthLayer = Layer.succeed(ClockifyAuth, {
|
|
94
|
+
getConfig: Effect.succeed({
|
|
95
|
+
apiKey: Redacted.make("key"),
|
|
96
|
+
workspaceId: WORKSPACE_ID,
|
|
97
|
+
userId: USER_ID,
|
|
98
|
+
baseUrl: "https://api.clockify.me/api"
|
|
99
|
+
}),
|
|
100
|
+
save: () => Effect.void,
|
|
101
|
+
isConfigured: Effect.succeed(true)
|
|
102
|
+
});
|
|
103
|
+
const MockConfigLayer = Layer.succeed(ConfigService, {
|
|
104
|
+
get: Effect.succeed({
|
|
105
|
+
defaultJql: "",
|
|
106
|
+
refreshInterval: 30,
|
|
107
|
+
projectMap: {},
|
|
108
|
+
workspaceId: null,
|
|
109
|
+
defaultProjectId: null,
|
|
110
|
+
defaultProjectName: null,
|
|
111
|
+
defaultBillable: true
|
|
112
|
+
}),
|
|
113
|
+
set: () => Effect.void,
|
|
114
|
+
configDir: Effect.succeed("/tmp/.jcf")
|
|
115
|
+
});
|
|
116
|
+
let writtenStates = [];
|
|
117
|
+
let cleared = false;
|
|
118
|
+
const MockStateWriterLayer = Layer.succeed(StateWriter, {
|
|
119
|
+
write: (state) => Effect.sync(() => {
|
|
120
|
+
writtenStates.push(state);
|
|
121
|
+
}),
|
|
122
|
+
read: Effect.succeed({
|
|
123
|
+
active: false,
|
|
124
|
+
ticketKey: null,
|
|
125
|
+
summary: null,
|
|
126
|
+
project: null,
|
|
127
|
+
startedAt: null,
|
|
128
|
+
startedAt_unix: null,
|
|
129
|
+
elapsed: 0,
|
|
130
|
+
clockifyEntryId: null
|
|
131
|
+
}),
|
|
132
|
+
clear: Effect.sync(() => {
|
|
133
|
+
cleared = true;
|
|
134
|
+
})
|
|
135
|
+
});
|
|
136
|
+
const MockJiraAuthLayer = Layer.succeed(JiraAuth, {
|
|
137
|
+
configure: () => Effect.void,
|
|
138
|
+
isConfigured: () => Effect.succeed(true),
|
|
139
|
+
login: () => Effect.void,
|
|
140
|
+
logout: () => Effect.void,
|
|
141
|
+
getAccessToken: () => Effect.succeed(Redacted.make("jira-token")),
|
|
142
|
+
getCloudId: () => Effect.succeed("cloud-1"),
|
|
143
|
+
getSiteUrl: () => Effect.succeed("https://test.atlassian.net"),
|
|
144
|
+
getCurrentUser: () => Effect.succeed(null),
|
|
145
|
+
isLoggedIn: () => Effect.succeed(true)
|
|
146
|
+
});
|
|
147
|
+
// Mock HttpClient that returns 201 for Jira worklog POST
|
|
148
|
+
const MockHttpClientLayer = Layer.succeed(HttpClient.HttpClient, HttpClient.make((request) => Effect.succeed(HttpClientResponse.fromWeb(request, new Response(JSON.stringify({ id: "wl-1" }), {
|
|
149
|
+
status: 201,
|
|
150
|
+
headers: { "content-type": "application/json" }
|
|
151
|
+
})))));
|
|
152
|
+
const TestLayer = timerLayer.pipe(Layer.provide(MockClockifyLayer), Layer.provide(MockJiraApiClientLayer), Layer.provide(MockClockifyAuthLayer), Layer.provide(MockConfigLayer), Layer.provide(MockStateWriterLayer), Layer.provide(MockJiraAuthLayer), Layer.provide(MockHttpClientLayer));
|
|
153
|
+
// ---------------------------------------------------------------------------
|
|
154
|
+
// Tests
|
|
155
|
+
// ---------------------------------------------------------------------------
|
|
156
|
+
describe("TimerService", () => {
|
|
157
|
+
describe("state transitions", () => {
|
|
158
|
+
// Initial state must be idle — ensures no stale state from previous sessions
|
|
159
|
+
it.effect("starts idle", () => Effect.gen(function* () {
|
|
160
|
+
resetCaptures();
|
|
161
|
+
writtenStates = [];
|
|
162
|
+
cleared = false;
|
|
163
|
+
const svc = yield* TimerService;
|
|
164
|
+
const state = yield* SubscriptionRef.get(svc.state);
|
|
165
|
+
expect(state.active).toBe(false);
|
|
166
|
+
expect(state.ticketKey).toBeNull();
|
|
167
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
168
|
+
// Core lifecycle: start creates Clockify entry + writes state file + updates SubscriptionRef
|
|
169
|
+
it.effect("idle -> active on start", () => Effect.gen(function* () {
|
|
170
|
+
resetCaptures();
|
|
171
|
+
writtenStates = [];
|
|
172
|
+
cleared = false;
|
|
173
|
+
const svc = yield* TimerService;
|
|
174
|
+
yield* svc.start(makeTicket());
|
|
175
|
+
const state = yield* SubscriptionRef.get(svc.state);
|
|
176
|
+
expect(state.active).toBe(true);
|
|
177
|
+
expect(state.ticketKey).toBe("PROJ-123");
|
|
178
|
+
expect(state.summary).toBe("Fix the widget");
|
|
179
|
+
expect(state.startedViaJcf).toBe(true);
|
|
180
|
+
expect(state.clockifyEntryId).toBe("entry-1");
|
|
181
|
+
expect(createdEntries).toHaveLength(1);
|
|
182
|
+
expect(writtenStates.length).toBeGreaterThanOrEqual(1);
|
|
183
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
184
|
+
// Core lifecycle: stop updates Clockify entry + posts Jira worklog + clears state
|
|
185
|
+
it.effect("active -> stopped on stop", () => Effect.gen(function* () {
|
|
186
|
+
resetCaptures();
|
|
187
|
+
writtenStates = [];
|
|
188
|
+
cleared = false;
|
|
189
|
+
const svc = yield* TimerService;
|
|
190
|
+
yield* svc.start(makeTicket());
|
|
191
|
+
const result = yield* svc.stop();
|
|
192
|
+
expect(result.clockifyLogged).toBe(true);
|
|
193
|
+
expect(result.jiraWorklogLogged).toBe(true);
|
|
194
|
+
const state = yield* SubscriptionRef.get(svc.state);
|
|
195
|
+
expect(state.active).toBe(false);
|
|
196
|
+
expect(state.ticketKey).toBeNull();
|
|
197
|
+
expect(cleared).toBe(true);
|
|
198
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
199
|
+
// Stop without active timer must fail — prevents orphaned Clockify/Jira state
|
|
200
|
+
it.effect("stop with no active timer fails", () => Effect.gen(function* () {
|
|
201
|
+
resetCaptures();
|
|
202
|
+
const svc = yield* TimerService;
|
|
203
|
+
const exit = yield* svc.stop().pipe(Effect.flip);
|
|
204
|
+
expect(exit).toBeInstanceOf(TimerError);
|
|
205
|
+
expect(exit.message).toBe("No active timer");
|
|
206
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
207
|
+
});
|
|
208
|
+
describe("auto-stop", () => {
|
|
209
|
+
// Starting a new timer must stop the existing one first — prevents orphaned Clockify entries
|
|
210
|
+
it.effect("starting new timer auto-stops existing one", () => Effect.gen(function* () {
|
|
211
|
+
resetCaptures();
|
|
212
|
+
writtenStates = [];
|
|
213
|
+
cleared = false;
|
|
214
|
+
const svc = yield* TimerService;
|
|
215
|
+
yield* svc.start(makeTicket({ key: "PROJ-1", summary: "First" }));
|
|
216
|
+
expect(createdEntries).toHaveLength(1);
|
|
217
|
+
// Start a second timer — should auto-stop the first
|
|
218
|
+
yield* svc.start(makeTicket({ key: "PROJ-2", summary: "Second" }));
|
|
219
|
+
expect(createdEntries).toHaveLength(2);
|
|
220
|
+
// The auto-stop calls updateTimeEntry on the first entry
|
|
221
|
+
expect(updatedEntries).toHaveLength(1);
|
|
222
|
+
const state = yield* SubscriptionRef.get(svc.state);
|
|
223
|
+
expect(state.ticketKey).toBe("PROJ-2");
|
|
224
|
+
expect(state.summary).toBe("Second");
|
|
225
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
226
|
+
});
|
|
227
|
+
describe("60s Jira worklog floor", () => {
|
|
228
|
+
// Jira rejects worklogs <60s — verify near-instant timers still log successfully (floored to 60s)
|
|
229
|
+
it.effect("floors timeSpentSeconds to 60 for short durations", () => Effect.gen(function* () {
|
|
230
|
+
resetCaptures();
|
|
231
|
+
writtenStates = [];
|
|
232
|
+
cleared = false;
|
|
233
|
+
const svc = yield* TimerService;
|
|
234
|
+
yield* svc.start(makeTicket());
|
|
235
|
+
// Stop immediately — elapsed ~0ms, should still log with 60s floor
|
|
236
|
+
const result = yield* svc.stop();
|
|
237
|
+
expect(result.jiraWorklogLogged).toBe(true);
|
|
238
|
+
expect(result.clockifyLogged).toBe(true);
|
|
239
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
240
|
+
});
|
|
241
|
+
describe("discard", () => {
|
|
242
|
+
// Discard deletes Clockify entry and clears state — no Jira worklog should be created
|
|
243
|
+
it.effect("discards active timer without Jira worklog", () => Effect.gen(function* () {
|
|
244
|
+
resetCaptures();
|
|
245
|
+
writtenStates = [];
|
|
246
|
+
cleared = false;
|
|
247
|
+
const svc = yield* TimerService;
|
|
248
|
+
yield* svc.start(makeTicket());
|
|
249
|
+
yield* svc.discard;
|
|
250
|
+
const state = yield* SubscriptionRef.get(svc.state);
|
|
251
|
+
expect(state.active).toBe(false);
|
|
252
|
+
expect(deletedEntries).toHaveLength(1);
|
|
253
|
+
expect(deletedEntries[0].id).toBe("entry-1");
|
|
254
|
+
expect(cleared).toBe(true);
|
|
255
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
256
|
+
// Discard without active timer must fail — prevents accidental entry deletion
|
|
257
|
+
it.effect("discard with no active timer fails", () => Effect.gen(function* () {
|
|
258
|
+
resetCaptures();
|
|
259
|
+
const svc = yield* TimerService;
|
|
260
|
+
const exit = yield* svc.discard.pipe(Effect.flip);
|
|
261
|
+
expect(exit).toBeInstanceOf(TimerError);
|
|
262
|
+
expect(exit.message).toBe("No active timer to discard");
|
|
263
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
264
|
+
});
|
|
265
|
+
describe("detectRunning", () => {
|
|
266
|
+
// Detects externally-started Clockify timers with "[KEY] summary" format (jcf native format)
|
|
267
|
+
it.effect("parses bracket format [KEY] summary", () => Effect.gen(function* () {
|
|
268
|
+
resetCaptures();
|
|
269
|
+
writtenStates = [];
|
|
270
|
+
cleared = false;
|
|
271
|
+
const runningEntry = makeTimeEntry("ext-1", "[PROJ-42] Implement feature", new Date("2025-01-01T10:00:00Z"), "proj-id");
|
|
272
|
+
const clockifyWithRunning = {
|
|
273
|
+
...mockClockify,
|
|
274
|
+
getRunningTimer: () => Effect.succeed(runningEntry),
|
|
275
|
+
getProjects: () => Effect.succeed([{
|
|
276
|
+
id: "proj-id",
|
|
277
|
+
name: "MyProject",
|
|
278
|
+
color: "",
|
|
279
|
+
archived: false,
|
|
280
|
+
billable: true,
|
|
281
|
+
public: true,
|
|
282
|
+
workspaceId: WORKSPACE_ID,
|
|
283
|
+
note: ""
|
|
284
|
+
}])
|
|
285
|
+
};
|
|
286
|
+
const layer = timerLayer.pipe(Layer.provide(Layer.succeed(ClockifyApiClient, clockifyWithRunning)), Layer.provide(MockJiraApiClientLayer), Layer.provide(MockClockifyAuthLayer), Layer.provide(MockConfigLayer), Layer.provide(MockStateWriterLayer), Layer.provide(MockJiraAuthLayer), Layer.provide(MockHttpClientLayer));
|
|
287
|
+
const svc = yield* TimerService.pipe(Effect.provide(layer));
|
|
288
|
+
yield* svc.detectRunning.pipe(Effect.provide(layer));
|
|
289
|
+
// State was updated via the layer's SubscriptionRef, so read from svc
|
|
290
|
+
const state = yield* SubscriptionRef.get(svc.state);
|
|
291
|
+
expect(state.active).toBe(true);
|
|
292
|
+
expect(state.ticketKey).toBe("PROJ-42");
|
|
293
|
+
expect(state.summary).toBe("Implement feature");
|
|
294
|
+
expect(state.projectId).toBe("proj-id");
|
|
295
|
+
expect(state.projectName).toBe("MyProject");
|
|
296
|
+
expect(state.startedViaJcf).toBe(false);
|
|
297
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
298
|
+
// Also detects "KEY: summary" format — common when timers are started manually in Clockify
|
|
299
|
+
it.effect("parses colon format KEY: summary", () => Effect.gen(function* () {
|
|
300
|
+
resetCaptures();
|
|
301
|
+
writtenStates = [];
|
|
302
|
+
const runningEntry = makeTimeEntry("ext-2", "PROJ-99: Review PR", new Date("2025-01-01T10:00:00Z"));
|
|
303
|
+
const clockifyWithRunning = {
|
|
304
|
+
...mockClockify,
|
|
305
|
+
getRunningTimer: () => Effect.succeed(runningEntry)
|
|
306
|
+
};
|
|
307
|
+
const layer = timerLayer.pipe(Layer.provide(Layer.succeed(ClockifyApiClient, clockifyWithRunning)), Layer.provide(MockJiraApiClientLayer), Layer.provide(MockClockifyAuthLayer), Layer.provide(MockConfigLayer), Layer.provide(MockStateWriterLayer), Layer.provide(MockJiraAuthLayer), Layer.provide(MockHttpClientLayer));
|
|
308
|
+
const svc = yield* TimerService.pipe(Effect.provide(layer));
|
|
309
|
+
yield* svc.detectRunning.pipe(Effect.provide(layer));
|
|
310
|
+
const state = yield* SubscriptionRef.get(svc.state);
|
|
311
|
+
expect(state.active).toBe(true);
|
|
312
|
+
expect(state.ticketKey).toBe("PROJ-99");
|
|
313
|
+
expect(state.summary).toBe("Review PR");
|
|
314
|
+
expect(state.startedViaJcf).toBe(false);
|
|
315
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
316
|
+
// No running timer in Clockify must leave local state unchanged — polling should be safe no-op
|
|
317
|
+
it.effect("no running timer is a no-op", () => Effect.gen(function* () {
|
|
318
|
+
resetCaptures();
|
|
319
|
+
writtenStates = [];
|
|
320
|
+
const svc = yield* TimerService;
|
|
321
|
+
yield* svc.detectRunning;
|
|
322
|
+
const state = yield* SubscriptionRef.get(svc.state);
|
|
323
|
+
expect(state.active).toBe(false);
|
|
324
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
325
|
+
});
|
|
326
|
+
describe("stop options", () => {
|
|
327
|
+
// Stop options override timer state — used when TUI prompts for projectId/billable on stop
|
|
328
|
+
it.effect("stop merges projectId and billable from options", () => Effect.gen(function* () {
|
|
329
|
+
resetCaptures();
|
|
330
|
+
writtenStates = [];
|
|
331
|
+
cleared = false;
|
|
332
|
+
const svc = yield* TimerService;
|
|
333
|
+
yield* svc.start(makeTicket());
|
|
334
|
+
const result = yield* svc.stop({ projectId: "proj-override", billable: false });
|
|
335
|
+
expect(result.needsProjectId).toBe(false);
|
|
336
|
+
expect(result.needsBillable).toBe(false);
|
|
337
|
+
// Check that the updateTimeEntry was called with the overridden values
|
|
338
|
+
expect(updatedEntries).toHaveLength(1);
|
|
339
|
+
const params = updatedEntries[0].params;
|
|
340
|
+
expect(params.projectId).toBe("proj-override");
|
|
341
|
+
expect(params.billable).toBe(false);
|
|
342
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
343
|
+
// needsProjectId=true signals TUI to prompt user — must be true when no project resolved
|
|
344
|
+
it.effect("needsProjectId true when no projectId provided", () => Effect.gen(function* () {
|
|
345
|
+
resetCaptures();
|
|
346
|
+
writtenStates = [];
|
|
347
|
+
cleared = false;
|
|
348
|
+
const svc = yield* TimerService;
|
|
349
|
+
yield* svc.start(makeTicket());
|
|
350
|
+
const result = yield* svc.stop();
|
|
351
|
+
expect(result.needsProjectId).toBe(true);
|
|
352
|
+
}).pipe(Effect.provide(TestLayer)));
|
|
353
|
+
});
|
|
354
|
+
});
|
|
355
|
+
//# sourceMappingURL=TimerService.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"TimerService.test.js","sourceRoot":"","sources":["../../test/TimerService.test.ts"],"names":[],"mappings":"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"}
|