Set up comprehensive frontend testing infrastructure
- Install Jest for unit testing with React Testing Library - Install Playwright for end-to-end testing - Configure Jest with proper TypeScript support and module mapping - Create test setup files and utilities for both unit and e2e tests Components: * Jest configuration with coverage thresholds * Playwright configuration with browser automation * Unit tests for LoginForm, AuthContext, and useSocketIO hook * E2E tests for authentication, dashboard, and agents workflows * GitHub Actions workflow for automated testing * Mock data and API utilities for consistent testing * Test documentation with best practices Testing features: - Unit tests with 70% coverage threshold - E2E tests with API mocking and user journey testing - CI/CD integration for automated test runs - Cross-browser testing support with Playwright - Authentication system testing end-to-end 🚀 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
		
							
								
								
									
										18
									
								
								frontend/node_modules/sprintf-js/src/angular-sprintf.js
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								frontend/node_modules/sprintf-js/src/angular-sprintf.js
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| angular. | ||||
|     module("sprintf", []). | ||||
|     filter("sprintf", function() { | ||||
|         return function() { | ||||
|             return sprintf.apply(null, arguments) | ||||
|         } | ||||
|     }). | ||||
|     filter("fmt", ["$filter", function($filter) { | ||||
|         return $filter("sprintf") | ||||
|     }]). | ||||
|     filter("vsprintf", function() { | ||||
|         return function(format, argv) { | ||||
|             return vsprintf(format, argv) | ||||
|         } | ||||
|     }). | ||||
|     filter("vfmt", ["$filter", function($filter) { | ||||
|         return $filter("vsprintf") | ||||
|     }]) | ||||
							
								
								
									
										208
									
								
								frontend/node_modules/sprintf-js/src/sprintf.js
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										208
									
								
								frontend/node_modules/sprintf-js/src/sprintf.js
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,208 @@ | ||||
| (function(window) { | ||||
|     var re = { | ||||
|         not_string: /[^s]/, | ||||
|         number: /[diefg]/, | ||||
|         json: /[j]/, | ||||
|         not_json: /[^j]/, | ||||
|         text: /^[^\x25]+/, | ||||
|         modulo: /^\x25{2}/, | ||||
|         placeholder: /^\x25(?:([1-9]\d*)\$|\(([^\)]+)\))?(\+)?(0|'[^$])?(-)?(\d+)?(?:\.(\d+))?([b-gijosuxX])/, | ||||
|         key: /^([a-z_][a-z_\d]*)/i, | ||||
|         key_access: /^\.([a-z_][a-z_\d]*)/i, | ||||
|         index_access: /^\[(\d+)\]/, | ||||
|         sign: /^[\+\-]/ | ||||
|     } | ||||
|  | ||||
|     function sprintf() { | ||||
|         var key = arguments[0], cache = sprintf.cache | ||||
|         if (!(cache[key] && cache.hasOwnProperty(key))) { | ||||
|             cache[key] = sprintf.parse(key) | ||||
|         } | ||||
|         return sprintf.format.call(null, cache[key], arguments) | ||||
|     } | ||||
|  | ||||
|     sprintf.format = function(parse_tree, argv) { | ||||
|         var cursor = 1, tree_length = parse_tree.length, node_type = "", arg, output = [], i, k, match, pad, pad_character, pad_length, is_positive = true, sign = "" | ||||
|         for (i = 0; i < tree_length; i++) { | ||||
|             node_type = get_type(parse_tree[i]) | ||||
|             if (node_type === "string") { | ||||
|                 output[output.length] = parse_tree[i] | ||||
|             } | ||||
|             else if (node_type === "array") { | ||||
|                 match = parse_tree[i] // convenience purposes only | ||||
|                 if (match[2]) { // keyword argument | ||||
|                     arg = argv[cursor] | ||||
|                     for (k = 0; k < match[2].length; k++) { | ||||
|                         if (!arg.hasOwnProperty(match[2][k])) { | ||||
|                             throw new Error(sprintf("[sprintf] property '%s' does not exist", match[2][k])) | ||||
|                         } | ||||
|                         arg = arg[match[2][k]] | ||||
|                     } | ||||
|                 } | ||||
|                 else if (match[1]) { // positional argument (explicit) | ||||
|                     arg = argv[match[1]] | ||||
|                 } | ||||
|                 else { // positional argument (implicit) | ||||
|                     arg = argv[cursor++] | ||||
|                 } | ||||
|  | ||||
|                 if (get_type(arg) == "function") { | ||||
|                     arg = arg() | ||||
|                 } | ||||
|  | ||||
|                 if (re.not_string.test(match[8]) && re.not_json.test(match[8]) && (get_type(arg) != "number" && isNaN(arg))) { | ||||
|                     throw new TypeError(sprintf("[sprintf] expecting number but found %s", get_type(arg))) | ||||
|                 } | ||||
|  | ||||
|                 if (re.number.test(match[8])) { | ||||
|                     is_positive = arg >= 0 | ||||
|                 } | ||||
|  | ||||
|                 switch (match[8]) { | ||||
|                     case "b": | ||||
|                         arg = arg.toString(2) | ||||
|                     break | ||||
|                     case "c": | ||||
|                         arg = String.fromCharCode(arg) | ||||
|                     break | ||||
|                     case "d": | ||||
|                     case "i": | ||||
|                         arg = parseInt(arg, 10) | ||||
|                     break | ||||
|                     case "j": | ||||
|                         arg = JSON.stringify(arg, null, match[6] ? parseInt(match[6]) : 0) | ||||
|                     break | ||||
|                     case "e": | ||||
|                         arg = match[7] ? arg.toExponential(match[7]) : arg.toExponential() | ||||
|                     break | ||||
|                     case "f": | ||||
|                         arg = match[7] ? parseFloat(arg).toFixed(match[7]) : parseFloat(arg) | ||||
|                     break | ||||
|                     case "g": | ||||
|                         arg = match[7] ? parseFloat(arg).toPrecision(match[7]) : parseFloat(arg) | ||||
|                     break | ||||
|                     case "o": | ||||
|                         arg = arg.toString(8) | ||||
|                     break | ||||
|                     case "s": | ||||
|                         arg = ((arg = String(arg)) && match[7] ? arg.substring(0, match[7]) : arg) | ||||
|                     break | ||||
|                     case "u": | ||||
|                         arg = arg >>> 0 | ||||
|                     break | ||||
|                     case "x": | ||||
|                         arg = arg.toString(16) | ||||
|                     break | ||||
|                     case "X": | ||||
|                         arg = arg.toString(16).toUpperCase() | ||||
|                     break | ||||
|                 } | ||||
|                 if (re.json.test(match[8])) { | ||||
|                     output[output.length] = arg | ||||
|                 } | ||||
|                 else { | ||||
|                     if (re.number.test(match[8]) && (!is_positive || match[3])) { | ||||
|                         sign = is_positive ? "+" : "-" | ||||
|                         arg = arg.toString().replace(re.sign, "") | ||||
|                     } | ||||
|                     else { | ||||
|                         sign = "" | ||||
|                     } | ||||
|                     pad_character = match[4] ? match[4] === "0" ? "0" : match[4].charAt(1) : " " | ||||
|                     pad_length = match[6] - (sign + arg).length | ||||
|                     pad = match[6] ? (pad_length > 0 ? str_repeat(pad_character, pad_length) : "") : "" | ||||
|                     output[output.length] = match[5] ? sign + arg + pad : (pad_character === "0" ? sign + pad + arg : pad + sign + arg) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         return output.join("") | ||||
|     } | ||||
|  | ||||
|     sprintf.cache = {} | ||||
|  | ||||
|     sprintf.parse = function(fmt) { | ||||
|         var _fmt = fmt, match = [], parse_tree = [], arg_names = 0 | ||||
|         while (_fmt) { | ||||
|             if ((match = re.text.exec(_fmt)) !== null) { | ||||
|                 parse_tree[parse_tree.length] = match[0] | ||||
|             } | ||||
|             else if ((match = re.modulo.exec(_fmt)) !== null) { | ||||
|                 parse_tree[parse_tree.length] = "%" | ||||
|             } | ||||
|             else if ((match = re.placeholder.exec(_fmt)) !== null) { | ||||
|                 if (match[2]) { | ||||
|                     arg_names |= 1 | ||||
|                     var field_list = [], replacement_field = match[2], field_match = [] | ||||
|                     if ((field_match = re.key.exec(replacement_field)) !== null) { | ||||
|                         field_list[field_list.length] = field_match[1] | ||||
|                         while ((replacement_field = replacement_field.substring(field_match[0].length)) !== "") { | ||||
|                             if ((field_match = re.key_access.exec(replacement_field)) !== null) { | ||||
|                                 field_list[field_list.length] = field_match[1] | ||||
|                             } | ||||
|                             else if ((field_match = re.index_access.exec(replacement_field)) !== null) { | ||||
|                                 field_list[field_list.length] = field_match[1] | ||||
|                             } | ||||
|                             else { | ||||
|                                 throw new SyntaxError("[sprintf] failed to parse named argument key") | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                     else { | ||||
|                         throw new SyntaxError("[sprintf] failed to parse named argument key") | ||||
|                     } | ||||
|                     match[2] = field_list | ||||
|                 } | ||||
|                 else { | ||||
|                     arg_names |= 2 | ||||
|                 } | ||||
|                 if (arg_names === 3) { | ||||
|                     throw new Error("[sprintf] mixing positional and named placeholders is not (yet) supported") | ||||
|                 } | ||||
|                 parse_tree[parse_tree.length] = match | ||||
|             } | ||||
|             else { | ||||
|                 throw new SyntaxError("[sprintf] unexpected placeholder") | ||||
|             } | ||||
|             _fmt = _fmt.substring(match[0].length) | ||||
|         } | ||||
|         return parse_tree | ||||
|     } | ||||
|  | ||||
|     var vsprintf = function(fmt, argv, _argv) { | ||||
|         _argv = (argv || []).slice(0) | ||||
|         _argv.splice(0, 0, fmt) | ||||
|         return sprintf.apply(null, _argv) | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * helpers | ||||
|      */ | ||||
|     function get_type(variable) { | ||||
|         return Object.prototype.toString.call(variable).slice(8, -1).toLowerCase() | ||||
|     } | ||||
|  | ||||
|     function str_repeat(input, multiplier) { | ||||
|         return Array(multiplier + 1).join(input) | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * export to either browser or node.js | ||||
|      */ | ||||
|     if (typeof exports !== "undefined") { | ||||
|         exports.sprintf = sprintf | ||||
|         exports.vsprintf = vsprintf | ||||
|     } | ||||
|     else { | ||||
|         window.sprintf = sprintf | ||||
|         window.vsprintf = vsprintf | ||||
|  | ||||
|         if (typeof define === "function" && define.amd) { | ||||
|             define(function() { | ||||
|                 return { | ||||
|                     sprintf: sprintf, | ||||
|                     vsprintf: vsprintf | ||||
|                 } | ||||
|             }) | ||||
|         } | ||||
|     } | ||||
| })(typeof window === "undefined" ? this : window); | ||||
		Reference in New Issue
	
	Block a user
	 anthonyrawlins
					anthonyrawlins