test_utilities.py 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. #!/usr/bin/env python3
  2. """
  3. Tests for cli/utils.py utility functions
  4. """
  5. import unittest
  6. import tempfile
  7. import os
  8. import zipfile
  9. from pathlib import Path
  10. import sys
  11. from skill_seekers.cli.utils import (
  12. has_api_key,
  13. get_api_key,
  14. get_upload_url,
  15. format_file_size,
  16. validate_skill_directory,
  17. validate_zip_file,
  18. print_upload_instructions
  19. )
  20. class TestAPIKeyFunctions(unittest.TestCase):
  21. """Test API key utility functions"""
  22. def setUp(self):
  23. """Store original API key state"""
  24. self.original_api_key = os.environ.get('ANTHROPIC_API_KEY')
  25. def tearDown(self):
  26. """Restore original API key state"""
  27. if self.original_api_key:
  28. os.environ['ANTHROPIC_API_KEY'] = self.original_api_key
  29. elif 'ANTHROPIC_API_KEY' in os.environ:
  30. del os.environ['ANTHROPIC_API_KEY']
  31. def test_has_api_key_when_set(self):
  32. """Test has_api_key returns True when key is set"""
  33. os.environ['ANTHROPIC_API_KEY'] = 'sk-ant-test-key'
  34. self.assertTrue(has_api_key())
  35. def test_has_api_key_when_not_set(self):
  36. """Test has_api_key returns False when key is not set"""
  37. if 'ANTHROPIC_API_KEY' in os.environ:
  38. del os.environ['ANTHROPIC_API_KEY']
  39. self.assertFalse(has_api_key())
  40. def test_has_api_key_when_empty_string(self):
  41. """Test has_api_key returns False when key is empty string"""
  42. os.environ['ANTHROPIC_API_KEY'] = ''
  43. self.assertFalse(has_api_key())
  44. def test_has_api_key_when_whitespace_only(self):
  45. """Test has_api_key returns False when key is whitespace"""
  46. os.environ['ANTHROPIC_API_KEY'] = ' '
  47. self.assertFalse(has_api_key())
  48. def test_get_api_key_returns_key(self):
  49. """Test get_api_key returns the actual key"""
  50. os.environ['ANTHROPIC_API_KEY'] = 'sk-ant-test-key'
  51. self.assertEqual(get_api_key(), 'sk-ant-test-key')
  52. def test_get_api_key_returns_none_when_not_set(self):
  53. """Test get_api_key returns None when not set"""
  54. if 'ANTHROPIC_API_KEY' in os.environ:
  55. del os.environ['ANTHROPIC_API_KEY']
  56. self.assertIsNone(get_api_key())
  57. def test_get_api_key_strips_whitespace(self):
  58. """Test get_api_key strips whitespace from key"""
  59. os.environ['ANTHROPIC_API_KEY'] = ' sk-ant-test-key '
  60. self.assertEqual(get_api_key(), 'sk-ant-test-key')
  61. class TestGetUploadURL(unittest.TestCase):
  62. """Test get_upload_url function"""
  63. def test_get_upload_url_returns_correct_url(self):
  64. """Test get_upload_url returns the correct Claude skills URL"""
  65. url = get_upload_url()
  66. self.assertEqual(url, "https://claude.ai/skills")
  67. def test_get_upload_url_returns_string(self):
  68. """Test get_upload_url returns a string"""
  69. url = get_upload_url()
  70. self.assertIsInstance(url, str)
  71. class TestFormatFileSize(unittest.TestCase):
  72. """Test format_file_size function"""
  73. def test_format_bytes_below_1kb(self):
  74. """Test formatting bytes below 1 KB"""
  75. self.assertEqual(format_file_size(500), "500 bytes")
  76. self.assertEqual(format_file_size(1023), "1023 bytes")
  77. def test_format_kilobytes(self):
  78. """Test formatting KB sizes"""
  79. self.assertEqual(format_file_size(1024), "1.0 KB")
  80. self.assertEqual(format_file_size(1536), "1.5 KB")
  81. self.assertEqual(format_file_size(10240), "10.0 KB")
  82. def test_format_megabytes(self):
  83. """Test formatting MB sizes"""
  84. self.assertEqual(format_file_size(1048576), "1.0 MB")
  85. self.assertEqual(format_file_size(1572864), "1.5 MB")
  86. self.assertEqual(format_file_size(10485760), "10.0 MB")
  87. def test_format_zero_bytes(self):
  88. """Test formatting zero bytes"""
  89. self.assertEqual(format_file_size(0), "0 bytes")
  90. def test_format_large_files(self):
  91. """Test formatting large file sizes"""
  92. # 100 MB
  93. self.assertEqual(format_file_size(104857600), "100.0 MB")
  94. # 1 GB (still shows as MB)
  95. self.assertEqual(format_file_size(1073741824), "1024.0 MB")
  96. class TestValidateSkillDirectory(unittest.TestCase):
  97. """Test validate_skill_directory function"""
  98. def test_valid_skill_directory(self):
  99. """Test validation of valid skill directory"""
  100. with tempfile.TemporaryDirectory() as tmpdir:
  101. skill_dir = Path(tmpdir) / "test-skill"
  102. skill_dir.mkdir()
  103. (skill_dir / "SKILL.md").write_text("# Test Skill")
  104. is_valid, error = validate_skill_directory(skill_dir)
  105. self.assertTrue(is_valid)
  106. self.assertIsNone(error)
  107. def test_nonexistent_directory(self):
  108. """Test validation of nonexistent directory"""
  109. is_valid, error = validate_skill_directory("/nonexistent/path")
  110. self.assertFalse(is_valid)
  111. self.assertIn("not found", error.lower())
  112. def test_file_instead_of_directory(self):
  113. """Test validation when path is a file"""
  114. with tempfile.NamedTemporaryFile() as tmpfile:
  115. is_valid, error = validate_skill_directory(tmpfile.name)
  116. self.assertFalse(is_valid)
  117. self.assertIn("not a directory", error.lower())
  118. def test_directory_without_skill_md(self):
  119. """Test validation of directory without SKILL.md"""
  120. with tempfile.TemporaryDirectory() as tmpdir:
  121. is_valid, error = validate_skill_directory(tmpdir)
  122. self.assertFalse(is_valid)
  123. self.assertIn("SKILL.md not found", error)
  124. class TestValidateZipFile(unittest.TestCase):
  125. """Test validate_zip_file function"""
  126. def test_valid_zip_file(self):
  127. """Test validation of valid .zip file"""
  128. with tempfile.TemporaryDirectory() as tmpdir:
  129. zip_path = Path(tmpdir) / "test-skill.zip"
  130. # Create a real zip file
  131. with zipfile.ZipFile(zip_path, 'w') as zf:
  132. zf.writestr("SKILL.md", "# Test")
  133. is_valid, error = validate_zip_file(zip_path)
  134. self.assertTrue(is_valid)
  135. self.assertIsNone(error)
  136. def test_nonexistent_file(self):
  137. """Test validation of nonexistent file"""
  138. is_valid, error = validate_zip_file("/nonexistent/file.zip")
  139. self.assertFalse(is_valid)
  140. self.assertIn("not found", error.lower())
  141. def test_directory_instead_of_file(self):
  142. """Test validation when path is a directory"""
  143. with tempfile.TemporaryDirectory() as tmpdir:
  144. is_valid, error = validate_zip_file(tmpdir)
  145. self.assertFalse(is_valid)
  146. self.assertIn("not a file", error.lower())
  147. def test_wrong_extension(self):
  148. """Test validation of file with wrong extension"""
  149. with tempfile.NamedTemporaryFile(suffix='.txt') as tmpfile:
  150. is_valid, error = validate_zip_file(tmpfile.name)
  151. self.assertFalse(is_valid)
  152. self.assertIn("not a .zip file", error.lower())
  153. class TestPrintUploadInstructions(unittest.TestCase):
  154. """Test print_upload_instructions function"""
  155. def test_print_upload_instructions_runs(self):
  156. """Test that print_upload_instructions executes without error"""
  157. with tempfile.TemporaryDirectory() as tmpdir:
  158. zip_path = Path(tmpdir) / "test.zip"
  159. zip_path.write_text("")
  160. # Should not raise exception
  161. try:
  162. print_upload_instructions(zip_path)
  163. except Exception as e:
  164. self.fail(f"print_upload_instructions raised {e}")
  165. def test_print_upload_instructions_accepts_string_path(self):
  166. """Test print_upload_instructions accepts string path"""
  167. with tempfile.TemporaryDirectory() as tmpdir:
  168. zip_path = str(Path(tmpdir) / "test.zip")
  169. Path(zip_path).write_text("")
  170. try:
  171. print_upload_instructions(zip_path)
  172. except Exception as e:
  173. self.fail(f"print_upload_instructions raised {e}")
  174. if __name__ == '__main__':
  175. unittest.main()