ASONGHP commited on
Commit
a409a9a
·
verified ·
1 Parent(s): 82873fc

Update hf.js

Browse files
Files changed (1) hide show
  1. hf.js +89 -512
hf.js CHANGED
@@ -223,13 +223,11 @@ app.get('/', (req, res) => {
223
  --text-secondary: #666666;
224
  --border-color: #e0e0e0;
225
  --accent-color: #5D5CDE;
226
- --accent-light: #7D7CED;
227
  --success-bg: #e1f5e1;
228
  --success-border: #c3e6cb;
229
  --warning-bg: #fff3cd;
230
  --warning-border: #ffeeba;
231
- --code-bg: #f5f5f5;
232
- --shadow-color: rgba(0,0,0,0.1);
233
  }
234
 
235
  [data-theme="dark"] {
@@ -240,58 +238,29 @@ app.get('/', (req, res) => {
240
  --text-secondary: #b0b0b0;
241
  --border-color: #444444;
242
  --accent-color: #7D7CED;
243
- --accent-light: #8D8CF0;
244
  --success-bg: #1e3a1e;
245
  --success-border: #2a5a2a;
246
  --warning-bg: #3a3018;
247
  --warning-border: #5a4820;
248
- --code-bg: #2d2d2d;
249
- --shadow-color: rgba(0,0,0,0.3);
250
- }
251
-
252
- * {
253
- box-sizing: border-box;
254
- margin: 0;
255
- padding: 0;
256
  }
257
 
258
  body {
259
  font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
260
- color: var(--text-primary);
261
- background-color: var(--bg-primary);
262
- max-width: 1200px;
263
  margin: 0 auto;
264
  padding: 20px;
265
  line-height: 1.6;
266
- transition: background-color 0.3s, color 0.3s;
267
- }
268
-
269
- h1, h2, h3, h4 {
270
- margin-top: 1.5rem;
271
- margin-bottom: 1rem;
272
  color: var(--text-primary);
273
- }
274
-
275
- p {
276
- margin-bottom: 1rem;
277
- }
278
-
279
- a {
280
- color: var(--accent-color);
281
- text-decoration: none;
282
- }
283
-
284
- a:hover {
285
- color: var(--accent-light);
286
- text-decoration: underline;
287
  }
288
 
289
  .container {
290
  background: var(--bg-secondary);
291
  border-radius: 10px;
292
- padding: 30px;
293
- box-shadow: 0 4px 16px var(--shadow-color);
294
- margin-bottom: 30px;
295
  }
296
 
297
  .header {
@@ -299,8 +268,6 @@ app.get('/', (req, res) => {
299
  justify-content: space-between;
300
  align-items: center;
301
  margin-bottom: 20px;
302
- padding-bottom: 20px;
303
- border-bottom: 1px solid var(--border-color);
304
  }
305
 
306
  .theme-toggle {
@@ -308,15 +275,12 @@ app.get('/', (req, res) => {
308
  align-items: center;
309
  }
310
 
311
- .theme-toggle-label {
312
- margin-right: 10px;
313
- }
314
-
315
  .switch {
316
  position: relative;
317
  display: inline-block;
318
  width: 50px;
319
  height: 24px;
 
320
  }
321
 
322
  .switch input {
@@ -349,32 +313,19 @@ app.get('/', (req, res) => {
349
  border-radius: 50%;
350
  }
351
 
352
- input:checked + .slider {
353
- background-color: var(--accent-light);
354
- }
355
-
356
  input:checked + .slider:before {
357
  transform: translateX(26px);
358
  }
359
 
360
  .info-item {
361
- margin-bottom: 16px;
362
- padding: 10px;
363
- background: var(--bg-secondary);
364
- border-radius: 6px;
365
- border: 1px solid var(--border-color);
366
- }
367
-
368
- .info-item strong {
369
- display: block;
370
- margin-bottom: 5px;
371
  }
372
 
373
  .status {
374
  background: ${proxyConfig ? 'var(--success-bg)' : 'var(--warning-bg)'};
375
- padding: 15px;
376
- border-radius: 8px;
377
- margin: 20px 0;
378
  border: 1px solid ${proxyConfig ? 'var(--success-border)' : 'var(--warning-border)'};
379
  }
380
 
@@ -386,154 +337,39 @@ app.get('/', (req, res) => {
386
 
387
  .model-list {
388
  display: grid;
389
- grid-template-columns: repeat(auto-fill, minmax(180px, 1fr));
390
- gap: 12px;
391
  margin-top: 15px;
392
  }
393
 
394
  .model-item {
395
  background: var(--bg-tertiary);
396
- padding: 10px 15px;
397
- border-radius: 6px;
398
  font-size: 0.9em;
399
- transition: all 0.2s;
400
- cursor: pointer;
401
- border: 1px solid var(--border-color);
402
- }
403
-
404
- .model-item:hover {
405
- transform: translateY(-2px);
406
- box-shadow: 0 4px 8px var(--shadow-color);
407
  }
408
 
409
  .guide-section {
410
- margin-top: 40px;
411
- }
412
-
413
- .step {
414
- margin-bottom: 20px;
415
- padding: 15px;
416
- background: var(--bg-secondary);
417
- border-radius: 8px;
418
- border-left: 4px solid var(--accent-color);
419
- }
420
-
421
- .step h3 {
422
- margin-top: 0;
423
- }
424
-
425
- .code-block {
426
- background: var(--code-bg);
427
- border-radius: 6px;
428
- padding: 15px;
429
- margin: 15px 0;
430
- overflow-x: auto;
431
- font-family: 'SFMono-Regular', Consolas, 'Liberation Mono', Menlo, monospace;
432
- font-size: 14px;
433
- line-height: 1.5;
434
- border: 1px solid var(--border-color);
435
- }
436
-
437
- .tab-container {
438
  margin-top: 20px;
 
 
439
  }
440
 
441
- .tabs {
442
- display: flex;
443
- border-bottom: 1px solid var(--border-color);
444
- }
445
-
446
- .tab {
447
- padding: 10px 20px;
448
- cursor: pointer;
449
- background: var(--bg-secondary);
450
- border: 1px solid var(--border-color);
451
- border-bottom: none;
452
- border-radius: 6px 6px 0 0;
453
- margin-right: 5px;
454
- }
455
-
456
- .tab.active {
457
- background: var(--accent-color);
458
- color: white;
459
- border-color: var(--accent-color);
460
- }
461
-
462
- .tab-content {
463
- display: none;
464
- padding: 20px;
465
- background: var(--bg-secondary);
466
- border: 1px solid var(--border-color);
467
- border-top: none;
468
- border-radius: 0 0 6px 6px;
469
- }
470
-
471
- .tab-content.active {
472
- display: block;
473
- }
474
-
475
- .api-tester {
476
- margin-top: 30px;
477
- }
478
-
479
- .form-group {
480
  margin-bottom: 15px;
481
- }
482
-
483
- .form-group label {
484
- display: block;
485
- margin-bottom: 5px;
486
- font-weight: bold;
487
- }
488
-
489
- .form-control {
490
- width: 100%;
491
  padding: 10px;
492
- border: 1px solid var(--border-color);
493
- border-radius: 4px;
494
- background: var(--bg-primary);
495
- color: var(--text-primary);
496
- font-size: 16px;
497
- }
498
-
499
- textarea.form-control {
500
- min-height: 100px;
501
- font-family: 'SFMono-Regular', Consolas, 'Liberation Mono', Menlo, monospace;
502
- }
503
-
504
- .btn {
505
- padding: 10px 20px;
506
- background: var(--accent-color);
507
- color: white;
508
- border: none;
509
- border-radius: 4px;
510
- cursor: pointer;
511
- font-size: 16px;
512
- transition: background 0.2s;
513
- }
514
-
515
- .btn:hover {
516
- background: var(--accent-light);
517
- }
518
-
519
- .response-container {
520
- margin-top: 20px;
521
- padding: 15px;
522
  background: var(--bg-secondary);
523
- border-radius: 6px;
524
- border: 1px solid var(--border-color);
525
- max-height: 300px;
526
- overflow-y: auto;
527
  }
528
 
529
- @media (max-width: 768px) {
530
- .container {
531
- padding: 20px;
532
- }
533
-
534
- .model-list {
535
- grid-template-columns: repeat(auto-fill, minmax(150px, 1fr));
536
- }
537
  }
538
  </style>
539
  </head>
@@ -542,7 +378,7 @@ app.get('/', (req, res) => {
542
  <div class="header">
543
  <h1>Cursor To OpenAI Server</h1>
544
  <div class="theme-toggle">
545
- <span class="theme-toggle-label">Dark Mode</span>
546
  <label class="switch">
547
  <input type="checkbox" id="theme-toggle">
548
  <span class="slider"></span>
@@ -569,252 +405,52 @@ app.get('/', (req, res) => {
569
  <h3>Supported Models</h3>
570
  <div id="model-list" class="model-list">Loading...</div>
571
  </div>
572
- </div>
573
-
574
- <div class="container guide-section">
575
- <h2>Complete Guide to Cursor API Integration</h2>
576
 
577
- <div class="tab-container">
578
- <div class="tabs">
579
- <div class="tab active" data-tab="setup">Setup</div>
580
- <div class="tab" data-tab="auth">Authentication</div>
581
- <div class="tab" data-tab="usage">Usage Examples</div>
582
- <div class="tab" data-tab="test">API Tester</div>
583
- </div>
584
-
585
- <div class="tab-content active" id="setup">
586
- <h3>Setting Up Cursor API Connection</h3>
587
-
588
- <div class="step">
589
- <h3>Step 1: Install the Required Dependencies</h3>
590
- <p>Make sure you have Node.js installed, then install the necessary packages:</p>
591
- <div class="code-block">
592
- npm install axios
593
- </div>
594
- </div>
595
-
596
- <div class="step">
597
- <h3>Step 2: Configure Your Environment</h3>
598
- <p>Set up your environment with the Cursor API endpoint:</p>
599
- <div class="code-block">
600
- const CURSOR_API_URL = "http://localhost:7860/hf/v1"; // Or your server URL
601
- </div>
602
- </div>
603
-
604
- <div class="step">
605
- <h3>Step 3: Create a Helper Function</h3>
606
- <p>Create a helper function to interact with the API:</p>
607
- <div class="code-block">
608
- const axios = require('axios');
609
-
610
- async function queryCursorAPI(model, messages, options = {}) {
611
- try {
612
- const response = await axios.post(`${CURSOR_API_URL}/chat/completions`, {
613
- model: model,
614
- messages: messages,
615
- temperature: options.temperature || 0.7,
616
- max_tokens: options.max_tokens,
617
- stream: options.stream || false
618
- }, {
619
- headers: {
620
- 'Content-Type': 'application/json',
621
- 'Authorization': `Bearer ${YOUR_API_KEY}` // Your Cursor cookie
622
- }
623
- });
624
-
625
- return response.data;
626
- } catch (error) {
627
- console.error('Error querying Cursor API:', error.response?.data || error.message);
628
- throw error;
629
- }
630
- }
631
- </div>
632
- </div>
633
- </div>
634
 
635
- <div class="tab-content" id="auth">
636
- <h3>Authentication with Cursor API</h3>
637
-
638
- <div class="step">
639
- <h3>Step 1: Obtain Your Cursor Cookie</h3>
640
- <p>To authenticate with the Cursor API, you need to extract your Cursor cookie:</p>
641
- <ol>
642
- <li>Log in to <a href="https://cursor.so" target="_blank">Cursor.so</a></li>
643
- <li>Open Developer Tools (F12 or Right-click Inspect)</li>
644
- <li>Go to the Application tab</li>
645
- <li>Under Storage, select Cookies</li>
646
- <li>Find the cookie that starts with <code>user_...</code></li>
647
- <li>Copy this value - this is your API key</li>
648
- </ol>
649
- </div>
650
-
651
- <div class="step">
652
- <h3>Step 2: Use the Cookie as Your API Key</h3>
653
- <p>Include your Cursor cookie in the Authorization header:</p>
654
- <div class="code-block">
655
- headers: {
656
- 'Content-Type': 'application/json',
657
- 'Authorization': 'Bearer user_your_cookie_value_here'
658
- }
659
- </div>
660
- <p class="warning">Important: Treat your Cursor cookie as a secret. Never share it publicly or include it in client-side code.</p>
661
- </div>
662
-
663
- <div class="step">
664
- <h3>Step 3: Test Authentication</h3>
665
- <p>Verify your authentication is working:</p>
666
- <div class="code-block">
667
- async function testAuth() {
668
- try {
669
- const response = await queryCursorAPI('claude-3.5-sonnet', [
670
- { role: 'user', content: 'Hello, can you hear me?' }
671
- ]);
672
- console.log('Authentication successful!', response.choices[0].message.content);
673
- } catch (error) {
674
- console.error('Authentication failed:', error);
675
- }
676
- }
677
-
678
- testAuth();
679
- </div>
680
- </div>
681
  </div>
682
 
683
- <div class="tab-content" id="usage">
684
- <h3>Cursor API Usage Examples</h3>
685
-
686
- <div class="step">
687
- <h3>Basic Text Completion</h3>
688
- <div class="code-block">
689
- const messages = [
690
- { role: 'system', content: 'You are a helpful assistant.' },
691
- { role: 'user', content: 'What is the capital of France?' }
692
- ];
693
-
694
- const response = await queryCursorAPI('claude-3.5-sonnet', messages);
695
- console.log(response.choices[0].message.content);
696
- </div>
697
- </div>
698
-
699
- <div class="step">
700
- <h3>Streaming Response</h3>
701
- <div class="code-block">
702
- const axios = require('axios');
703
-
704
- async function streamResponse(model, messages) {
705
- try {
706
- const response = await axios.post(`${CURSOR_API_URL}/chat/completions`, {
707
- model: model,
708
- messages: messages,
709
- stream: true
710
- }, {
711
- headers: {
712
- 'Content-Type': 'application/json',
713
- 'Authorization': `Bearer ${YOUR_API_KEY}`
714
- },
715
- responseType: 'stream'
716
- });
717
-
718
- response.data.on('data', chunk => {
719
- const lines = chunk.toString().split('\\n').filter(line => line.trim() !== '');
720
-
721
- for (const line of lines) {
722
- if (line.includes('[DONE]')) return;
723
-
724
- if (line.startsWith('data:')) {
725
- const jsonStr = line.replace('data:', '').trim();
726
- try {
727
- const json = JSON.parse(jsonStr);
728
- if (json.choices?.[0]?.delta?.content) {
729
- process.stdout.write(json.choices[0].delta.content);
730
- }
731
- } catch (e) {
732
- // Skip invalid JSON
733
- }
734
- }
735
- }
736
- });
737
-
738
- } catch (error) {
739
- console.error('Streaming error:', error);
740
- }
741
- }
742
-
743
- // Usage
744
- streamResponse('gpt-4o', [{ role: 'user', content: 'Write a short story about a robot.' }]);
745
- </div>
746
- </div>
747
-
748
- <div class="step">
749
- <h3>Using Different Models</h3>
750
- <p>You can use any model from the supported models list:</p>
751
- <div class="code-block">
752
- // Using Claude 3.7 Sonnet
753
- const claudeResponse = await queryCursorAPI('claude-3.7-sonnet', [
754
- { role: 'user', content: 'Explain quantum computing briefly.' }
755
- ]);
756
-
757
- // Using GPT-4o
758
- const gptResponse = await queryCursorAPI('gpt-4o', [
759
- { role: 'user', content: 'What are the benefits of clean energy?' }
760
- ]);
761
-
762
- // Using the "thinking" variant for complex reasoning
763
- const thinkingResponse = await queryCursorAPI('claude-3.7-sonnet-thinking', [
764
- { role: 'user', content: 'Solve this math problem: If a² + b² = 25 and ab = 12, what is a + b?' }
765
- ]);
766
- </div>
767
- </div>
768
-
769
- <div class="step">
770
- <h3>Setting Custom Parameters</h3>
771
- <div class="code-block">
772
- const response = await queryCursorAPI('gpt-4', [
773
- { role: 'user', content: 'Generate a creative story idea.' }
774
- ], {
775
- temperature: 0.9, // Higher creativity
776
- max_tokens: 500 // Limit response length
777
- });
778
- </div>
779
- </div>
780
  </div>
781
 
782
- <div class="tab-content" id="test">
783
- <h3>Cursor API Tester</h3>
784
- <p>Use this tool to test API calls directly from the browser:</p>
785
-
786
- <div class="api-tester">
787
- <div class="form-group">
788
- <label for="api-key">API Key (Cursor Cookie)</label>
789
- <input type="password" id="api-key" class="form-control" placeholder="user_...">
790
- </div>
791
-
792
- <div class="form-group">
793
- <label for="model-select">Model</label>
794
- <select id="model-select" class="form-control">
795
- <option value="claude-3.7-sonnet">claude-3.7-sonnet</option>
796
- <option value="gpt-4o">gpt-4o</option>
797
- <option value="claude-3.5-sonnet">claude-3.5-sonnet</option>
798
- <option value="gpt-4">gpt-4</option>
799
- </select>
800
- </div>
801
-
802
- <div class="form-group">
803
- <label for="prompt-input">Prompt</label>
804
- <textarea id="prompt-input" class="form-control" placeholder="Enter your prompt here...">Hello, can you introduce yourself?</textarea>
805
- </div>
806
-
807
- <div class="form-group">
808
- <label for="temperature">Temperature</label>
809
- <input type="range" id="temperature" class="form-control" min="0" max="1" step="0.1" value="0.7">
810
- <span id="temperature-value">0.7</span>
811
- </div>
812
-
813
- <button id="submit-api" class="btn">Send Request</button>
814
-
815
- <div class="response-container">
816
- <pre id="api-response">Response will appear here...</pre>
817
- </div>
818
  </div>
819
  </div>
820
  </div>
@@ -824,9 +460,8 @@ const thinkingResponse = await queryCursorAPI('claude-3.7-sonnet-thinking', [
824
  // Theme toggling
825
  const themeToggle = document.getElementById('theme-toggle');
826
 
827
- // Check for saved theme preference or system preference
828
- if (localStorage.getItem('theme') === 'dark' ||
829
- (!localStorage.getItem('theme') && window.matchMedia('(prefers-color-scheme: dark)').matches)) {
830
  document.documentElement.setAttribute('data-theme', 'dark');
831
  themeToggle.checked = true;
832
  }
@@ -835,10 +470,19 @@ const thinkingResponse = await queryCursorAPI('claude-3.7-sonnet-thinking', [
835
  themeToggle.addEventListener('change', function() {
836
  if (this.checked) {
837
  document.documentElement.setAttribute('data-theme', 'dark');
838
- localStorage.setItem('theme', 'dark');
839
  } else {
840
- document.documentElement.setAttribute('data-theme', 'light');
841
- localStorage.setItem('theme', 'light');
 
 
 
 
 
 
 
 
 
 
842
  }
843
  });
844
 
@@ -847,6 +491,12 @@ const thinkingResponse = await queryCursorAPI('claude-3.7-sonnet-thinking', [
847
  const link = url.protocol + '//' + url.host + '/hf/v1';
848
  document.getElementById('endpoint-url').textContent = link;
849
 
 
 
 
 
 
 
850
  // Load models list
851
  fetch('/hf/v1/models')
852
  .then(response => response.json())
@@ -854,89 +504,16 @@ const thinkingResponse = await queryCursorAPI('claude-3.7-sonnet-thinking', [
854
  const modelListEl = document.getElementById('model-list');
855
  modelListEl.innerHTML = '';
856
 
857
- // Sort models alphabetically
858
- data.data.sort((a, b) => a.id.localeCompare(b.id));
859
-
860
  data.data.forEach(model => {
861
  const modelEl = document.createElement('div');
862
  modelEl.className = 'model-item';
863
  modelEl.textContent = model.id;
864
- modelEl.addEventListener('click', () => {
865
- if (document.getElementById('model-select')) {
866
- document.getElementById('model-select').value = model.id;
867
- }
868
- });
869
  modelListEl.appendChild(modelEl);
870
-
871
- // Also add to dropdown if it exists
872
- if (document.getElementById('model-select')) {
873
- const option = document.createElement('option');
874
- option.value = model.id;
875
- option.textContent = model.id;
876
- document.getElementById('model-select').appendChild(option);
877
- }
878
  });
879
  })
880
  .catch(err => {
881
  document.getElementById('model-list').textContent = 'Failed to load models: ' + err.message;
882
  });
883
-
884
- // Tab switching
885
- const tabs = document.querySelectorAll('.tab');
886
- tabs.forEach(tab => {
887
- tab.addEventListener('click', () => {
888
- // Remove active class from all tabs
889
- document.querySelectorAll('.tab').forEach(t => t.classList.remove('active'));
890
- document.querySelectorAll('.tab-content').forEach(c => c.classList.remove('active'));
891
-
892
- // Add active class to clicked tab
893
- tab.classList.add('active');
894
- document.getElementById(tab.dataset.tab).classList.add('active');
895
- });
896
- });
897
-
898
- // Temperature slider
899
- const temperatureSlider = document.getElementById('temperature');
900
- const temperatureValue = document.getElementById('temperature-value');
901
- if (temperatureSlider) {
902
- temperatureSlider.addEventListener('input', () => {
903
- temperatureValue.textContent = temperatureSlider.value;
904
- });
905
- }
906
-
907
- // API test functionality
908
- const submitApiBtn = document.getElementById('submit-api');
909
- if (submitApiBtn) {
910
- submitApiBtn.addEventListener('click', async () => {
911
- const apiKey = document.getElementById('api-key').value;
912
- const model = document.getElementById('model-select').value;
913
- const prompt = document.getElementById('prompt-input').value;
914
- const temperature = parseFloat(document.getElementById('temperature').value);
915
-
916
- const responseContainer = document.getElementById('api-response');
917
- responseContainer.textContent = 'Loading...';
918
-
919
- try {
920
- const response = await fetch(`${link}/chat/completions`, {
921
- method: 'POST',
922
- headers: {
923
- 'Content-Type': 'application/json',
924
- 'Authorization': `Bearer ${apiKey}`
925
- },
926
- body: JSON.stringify({
927
- model: model,
928
- messages: [{ role: 'user', content: prompt }],
929
- temperature: temperature
930
- })
931
- });
932
-
933
- const data = await response.json();
934
- responseContainer.textContent = JSON.stringify(data, null, 2);
935
- } catch (error) {
936
- responseContainer.textContent = 'Error: ' + (error.message || 'Unknown error');
937
- }
938
- });
939
- }
940
  </script>
941
  </body>
942
  </html>
 
223
  --text-secondary: #666666;
224
  --border-color: #e0e0e0;
225
  --accent-color: #5D5CDE;
 
226
  --success-bg: #e1f5e1;
227
  --success-border: #c3e6cb;
228
  --warning-bg: #fff3cd;
229
  --warning-border: #ffeeba;
230
+ --card-shadow: 0 2px 10px rgba(0,0,0,0.1);
 
231
  }
232
 
233
  [data-theme="dark"] {
 
238
  --text-secondary: #b0b0b0;
239
  --border-color: #444444;
240
  --accent-color: #7D7CED;
 
241
  --success-bg: #1e3a1e;
242
  --success-border: #2a5a2a;
243
  --warning-bg: #3a3018;
244
  --warning-border: #5a4820;
245
+ --card-shadow: 0 2px 10px rgba(0,0,0,0.3);
 
 
 
 
 
 
 
246
  }
247
 
248
  body {
249
  font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
250
+ max-width: 800px;
 
 
251
  margin: 0 auto;
252
  padding: 20px;
253
  line-height: 1.6;
254
+ background-color: var(--bg-primary);
 
 
 
 
 
255
  color: var(--text-primary);
256
+ transition: background-color 0.3s, color 0.3s;
 
 
 
 
 
 
 
 
 
 
 
 
 
257
  }
258
 
259
  .container {
260
  background: var(--bg-secondary);
261
  border-radius: 10px;
262
+ padding: 20px;
263
+ box-shadow: var(--card-shadow);
 
264
  }
265
 
266
  .header {
 
268
  justify-content: space-between;
269
  align-items: center;
270
  margin-bottom: 20px;
 
 
271
  }
272
 
273
  .theme-toggle {
 
275
  align-items: center;
276
  }
277
 
 
 
 
 
278
  .switch {
279
  position: relative;
280
  display: inline-block;
281
  width: 50px;
282
  height: 24px;
283
+ margin-left: 10px;
284
  }
285
 
286
  .switch input {
 
313
  border-radius: 50%;
314
  }
315
 
 
 
 
 
316
  input:checked + .slider:before {
317
  transform: translateX(26px);
318
  }
319
 
320
  .info-item {
321
+ margin-bottom: 10px;
 
 
 
 
 
 
 
 
 
322
  }
323
 
324
  .status {
325
  background: ${proxyConfig ? 'var(--success-bg)' : 'var(--warning-bg)'};
326
+ padding: 10px;
327
+ border-radius: 5px;
328
+ margin-top: 20px;
329
  border: 1px solid ${proxyConfig ? 'var(--success-border)' : 'var(--warning-border)'};
330
  }
331
 
 
337
 
338
  .model-list {
339
  display: grid;
340
+ grid-template-columns: repeat(auto-fill, minmax(200px, 1fr));
341
+ gap: 10px;
342
  margin-top: 15px;
343
  }
344
 
345
  .model-item {
346
  background: var(--bg-tertiary);
347
+ padding: 8px 12px;
348
+ border-radius: 4px;
349
  font-size: 0.9em;
 
 
 
 
 
 
 
 
350
  }
351
 
352
  .guide-section {
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
353
  margin-top: 20px;
354
+ border-top: 1px solid var(--border-color);
355
+ padding-top: 20px;
356
  }
357
 
358
+ .guide-step {
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
359
  margin-bottom: 15px;
 
 
 
 
 
 
 
 
 
 
360
  padding: 10px;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
361
  background: var(--bg-secondary);
362
+ border-radius: 5px;
363
+ border-left: 3px solid var(--accent-color);
 
 
364
  }
365
 
366
+ .code-example {
367
+ background-color: var(--bg-tertiary);
368
+ padding: 10px;
369
+ border-radius: 5px;
370
+ overflow-x: auto;
371
+ font-family: monospace;
372
+ margin: 10px 0;
 
373
  }
374
  </style>
375
  </head>
 
378
  <div class="header">
379
  <h1>Cursor To OpenAI Server</h1>
380
  <div class="theme-toggle">
381
+ Dark Mode
382
  <label class="switch">
383
  <input type="checkbox" id="theme-toggle">
384
  <span class="slider"></span>
 
405
  <h3>Supported Models</h3>
406
  <div id="model-list" class="model-list">Loading...</div>
407
  </div>
 
 
 
 
408
 
409
+ <div class="guide-section">
410
+ <h3>How to Connect to Cursor API</h3>
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
411
 
412
+ <div class="guide-step">
413
+ <h4>Step 1: Authentication</h4>
414
+ <p>To authenticate with the Cursor API, you need to obtain your Cursor cookie:</p>
415
+ <ol>
416
+ <li>Log in to <a href="https://cursor.so" target="_blank">Cursor.so</a></li>
417
+ <li>Open browser developer tools (F12)</li>
418
+ <li>Go to Application tab → Cookies</li>
419
+ <li>Find the cookie that starts with "user_..."</li>
420
+ <li>Use this value as your API key in the Authorization header</li>
421
+ </ol>
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
422
  </div>
423
 
424
+ <div class="guide-step">
425
+ <h4>Step 2: Set up your API client</h4>
426
+ <p>Configure your API client with the following settings:</p>
427
+ <ul>
428
+ <li>Base URL: <span id="endpoint-url-guide"></span></li>
429
+ <li>Headers: <code>Authorization: Bearer your_cursor_cookie</code></li>
430
+ <li>Content-Type: application/json</li>
431
+ </ul>
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
432
  </div>
433
 
434
+ <div class="guide-step">
435
+ <h4>Step 3: Make API requests</h4>
436
+ <p>Example request to generate a completion:</p>
437
+ <div class="code-example">
438
+ fetch('<span id="endpoint-url-code"></span>/chat/completions', {
439
+ method: 'POST',
440
+ headers: {
441
+ 'Content-Type': 'application/json',
442
+ 'Authorization': 'Bearer your_cursor_cookie'
443
+ },
444
+ body: JSON.stringify({
445
+ model: 'claude-3.7-sonnet',
446
+ messages: [
447
+ { role: 'user', content: 'Hello, who are you?' }
448
+ ],
449
+ temperature: 0.7
450
+ })
451
+ })
452
+ .then(response => response.json())
453
+ .then(data => console.log(data));
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
454
  </div>
455
  </div>
456
  </div>
 
460
  // Theme toggling
461
  const themeToggle = document.getElementById('theme-toggle');
462
 
463
+ // Check system preference
464
+ if (window.matchMedia && window.matchMedia('(prefers-color-scheme: dark)').matches) {
 
465
  document.documentElement.setAttribute('data-theme', 'dark');
466
  themeToggle.checked = true;
467
  }
 
470
  themeToggle.addEventListener('change', function() {
471
  if (this.checked) {
472
  document.documentElement.setAttribute('data-theme', 'dark');
 
473
  } else {
474
+ document.documentElement.removeAttribute('data-theme');
475
+ }
476
+ });
477
+
478
+ // Listen for system theme changes
479
+ window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', event => {
480
+ if (event.matches) {
481
+ document.documentElement.setAttribute('data-theme', 'dark');
482
+ themeToggle.checked = true;
483
+ } else {
484
+ document.documentElement.removeAttribute('data-theme');
485
+ themeToggle.checked = false;
486
  }
487
  });
488
 
 
491
  const link = url.protocol + '//' + url.host + '/hf/v1';
492
  document.getElementById('endpoint-url').textContent = link;
493
 
494
+ // Also update the guide URLs
495
+ const guideUrlElements = document.querySelectorAll('#endpoint-url-guide, #endpoint-url-code');
496
+ guideUrlElements.forEach(el => {
497
+ el.textContent = link;
498
+ });
499
+
500
  // Load models list
501
  fetch('/hf/v1/models')
502
  .then(response => response.json())
 
504
  const modelListEl = document.getElementById('model-list');
505
  modelListEl.innerHTML = '';
506
 
 
 
 
507
  data.data.forEach(model => {
508
  const modelEl = document.createElement('div');
509
  modelEl.className = 'model-item';
510
  modelEl.textContent = model.id;
 
 
 
 
 
511
  modelListEl.appendChild(modelEl);
 
 
 
 
 
 
 
 
512
  });
513
  })
514
  .catch(err => {
515
  document.getElementById('model-list').textContent = 'Failed to load models: ' + err.message;
516
  });
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
517
  </script>
518
  </body>
519
  </html>