Skip to content

Commit 22765e9

Browse files
Merge pull request #124 from danyeaw/logging
Enhanced Frontend Logging System
2 parents 1ec15f7 + 2573a04 commit 22765e9

18 files changed

+1656
-458
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,7 @@ coverage.xml
7171
*.cover
7272
.hypothesis/
7373
.pytest_cache/
74+
junit.xml
7475

7576
# Translations
7677
*.mo

CONTRIBUTING.md

+89
Original file line numberDiff line numberDiff line change
@@ -82,3 +82,92 @@ More information are provided within the [ui-tests](./ui-tests/README.md) README
8282
### Packaging the extension
8383

8484
See [RELEASE](RELEASE.md)
85+
86+
### Logging
87+
88+
The extension uses a configurable logging system that supports different
89+
verbosity levels. When developing or debugging, you can adjust the log level to
90+
see more or less detailed information.
91+
92+
#### Log Levels
93+
94+
The following log levels are available (in order of increasing verbosity):
95+
96+
- `NONE` (0): No logs
97+
- `ERROR` (1): Only error messages
98+
- `WARN` (2): Warnings and errors
99+
- `INFO` (3): Informational messages, warnings, and errors (default)
100+
- `DEBUG` (4): Debug messages and all above
101+
102+
#### Setting the Log Level During Development
103+
104+
There are several ways to change the log level during development:
105+
106+
1. **Using the JupyterLab Settings UI:**
107+
108+
- Open JupyterLab
109+
- Go to Settings → Settings Editor
110+
- Select "jupyter-fsspec" in the left sidebar
111+
- Set the `logLevel` value in the User Settings panel on the right:
112+
```json
113+
{
114+
"logLevel": "debug"
115+
}
116+
```
117+
- Click "Save Settings"
118+
119+
2. **Programmatically in the browser console:**
120+
121+
```javascript
122+
// Set to debug level
123+
window.jupyterFsspecLogConfig.setLevel(4);
124+
125+
// Set to error level only
126+
window.jupyterFsspecLogConfig.setLevel(1);
127+
```
128+
129+
3. **In your code during development:**
130+
131+
```typescript
132+
import { Logger } from './logger';
133+
134+
// Set log level for debugging
135+
Logger.setLevel(Logger.DEBUG);
136+
137+
// Create a logger with context
138+
const logger = Logger.getLogger('MyComponent');
139+
logger.debug('This is a debug message');
140+
```
141+
142+
#### Best Practices for Logging
143+
144+
When adding logging to your code:
145+
146+
1. Always use a contextual logger:
147+
148+
```typescript
149+
const logger = Logger.getLogger('ComponentName');
150+
```
151+
152+
2. Choose the appropriate log level:
153+
154+
- `error`: For failures that prevent functionality
155+
- `warn`: For issues that don't break functionality but are concerning
156+
- `info`: For important events users should know about
157+
- `debug`: For developer-focused details
158+
159+
3. Include relevant data for debugging:
160+
161+
```typescript
162+
logger.debug('Processing data', {
163+
count: items.length,
164+
firstItem: items[0]
165+
});
166+
```
167+
168+
4. Log at entry/exit points of significant operations:
169+
```typescript
170+
logger.info('Starting file upload...');
171+
// ... operation code ...
172+
logger.info('File upload completed');
173+
```

jest.config.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,6 @@ module.exports = {
2323
'!src/**/.ipynb_checkpoints/*'
2424
],
2525
coverageReporters: ['lcov', 'text'],
26-
testRegex: 'src/.*/.*.spec.ts[x]?$',
26+
testRegex: 'src/.*/.*.test.ts[x]?$',
2727
transformIgnorePatterns: [`/node_modules/(?!${esModules}).+`]
2828
};

schema/plugin.json

+11-3
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,17 @@
11
{
22
"jupyter.lab.shortcuts": [],
3-
"title": "jupyterFsspec",
4-
"description": "jupyterFsspec settings.",
3+
"title": "Jupyter Fsspec",
4+
"description": "Settings for the jupyter-fsspec extension.",
55
"type": "object",
6-
"properties": {},
6+
"properties": {
7+
"logLevel": {
8+
"type": "string",
9+
"title": "Log Level",
10+
"description": "Set the verbosity of logging (none, error, warn, info, debug)",
11+
"enum": ["none", "error", "warn", "info", "debug"],
12+
"default": "info"
13+
}
14+
},
715
"additionalProperties": false,
816
"jupyter.lab.menus": {
917
"main": [

src/FssFilesysContextMenu.ts

+54-10
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,18 @@
11
// Right-click/context menu for file items
22
import { INotebookTracker } from '@jupyterlab/notebook';
3+
import { Logger } from './logger';
34

45
export class FssFilesysContextMenu {
56
root: any;
67
clicked = false;
78
parentControl: any = null;
89
model: any;
910
notebookTracker: any;
11+
private readonly logger: Logger;
1012

1113
constructor(model: any, notebookTracker: INotebookTracker) {
14+
this.logger = Logger.getLogger('FssFilesysContextMenu');
15+
1216
const root = document.createElement('div');
1317
root.classList.add('jfss-tree-context-menu');
1418
this.root = root;
@@ -28,6 +32,7 @@ export class FssFilesysContextMenu {
2832
}
2933

3034
root.addEventListener('mouseleave', this.handleMouseExit.bind(this), false);
35+
this.logger.debug('Context menu initialized');
3136
}
3237

3338
createMenuItem(text: string, cssClass: string, contextType: string) {
@@ -41,6 +46,7 @@ export class FssFilesysContextMenu {
4146
menuItem.addEventListener('mouseleave', this.handleItemUnhover.bind(this));
4247

4348
this.root.appendChild(menuItem);
49+
this.logger.debug('Created menu item', { text, contextType });
4450

4551
return menuItem;
4652
}
@@ -54,7 +60,14 @@ export class FssFilesysContextMenu {
5460
if (protocol) {
5561
const canonical =
5662
protocol + '/' + this.root.dataset.fss.replace(/^\/+/, () => '');
63+
this.logger.debug('Generated path', { path: canonical });
5764
return canonical;
65+
} else {
66+
this.logger.warn('Failed to generate path', {
67+
reason: 'No protocol found',
68+
info: info
69+
});
70+
return undefined;
5871
}
5972
}
6073

@@ -65,14 +78,20 @@ export class FssFilesysContextMenu {
6578
navigator.clipboard.writeText(path).then(
6679
() => {
6780
// Success
68-
console.log('Copy path: ' + path);
81+
this.logger.info('Path copied to clipboard', { path });
6982
this.root.remove();
7083
},
71-
() => {
72-
console.log('Copy path failed: ' + path);
84+
error => {
85+
this.logger.error('Failed to copy path to clipboard', {
86+
path,
87+
error
88+
});
7389
this.root.remove();
7490
}
7591
);
92+
} else {
93+
this.logger.error('Cannot copy path', { reason: 'path is undefined' });
94+
this.root.remove();
7695
}
7796
}
7897

@@ -85,36 +104,60 @@ export class FssFilesysContextMenu {
85104
const cellContent = activeCell.model.sharedModel.getSource();
86105
const newCellContent = cellContent + '\n' + codeBlock;
87106
activeCell.model.sharedModel.setSource(newCellContent);
88-
console.log('Updated cell content to: ', newCellContent);
107+
this.logger.debug('Updated cell content', {
108+
oldLength: cellContent.length,
109+
newLength: newCellContent.length,
110+
notebookType: notebookPanel.content.model.type
111+
});
112+
} else {
113+
this.logger.warn('No active cell found in notebook', {
114+
notebookId: notebookPanel.id
115+
});
89116
}
117+
} else {
118+
this.logger.warn('No active notebook found');
90119
}
91120
}
121+
92122
copyOpenCodeBlock() {
93123
const path = this.copyPath();
94124

95125
if (path) {
96126
const openCodeBlock = `with fsspec.open("${path}", "rb") as f:\n ...`;
97127
navigator.clipboard.writeText(openCodeBlock).then(
98128
() => {
99-
console.log('Copied `open` code block');
100-
console.log(openCodeBlock);
129+
this.logger.info('Copied code snippet to clipboard', {
130+
operation: 'open',
131+
path
132+
});
133+
this.logger.debug('Code block content', { content: openCodeBlock });
101134
this.root.remove();
102135
},
103-
() => {
104-
console.log('Failed to copy `open` code block');
136+
error => {
137+
this.logger.error('Failed to copy code snippet to clipboard', {
138+
operation: 'open',
139+
path,
140+
error
141+
});
105142
this.root.remove();
106143
}
107144
);
108145

109146
this.insertCodeBlock(openCodeBlock);
110147
} else {
111-
console.log('Failed to copy `open` code block');
148+
this.logger.error('Failed to copy code snippet', {
149+
operation: 'open',
150+
reason: 'path not available'
151+
});
112152
this.root.remove();
113153
}
114154
}
115155

116156
handleItemClick(event: any) {
117-
// TODO multiple menu it
157+
this.logger.debug('Menu item clicked', {
158+
type: event.target.dataset.fssContextType
159+
});
160+
118161
if (event.target.dataset.fssContextType === 'copyPath') {
119162
this.copyPathToClipboard();
120163
} else if (event.target.dataset.fssContextType === 'copyOpenCodeBlock') {
@@ -134,6 +177,7 @@ export class FssFilesysContextMenu {
134177

135178
handleMouseExit(event: any) {
136179
event.preventDefault();
180+
this.logger.debug('Context menu closed', { reason: 'mouse exit' });
137181
this.root.remove();
138182
return false;
139183
}

0 commit comments

Comments
 (0)