feat: Production readiness improvements for WHOOSH council formation
Major security, observability, and configuration improvements:
## Security Hardening
- Implemented configurable CORS (no more wildcards)
- Added comprehensive auth middleware for admin endpoints
- Enhanced webhook HMAC validation
- Added input validation and rate limiting
- Security headers and CSP policies
## Configuration Management
- Made N8N webhook URL configurable (WHOOSH_N8N_BASE_URL)
- Replaced all hardcoded endpoints with environment variables
- Added feature flags for LLM vs heuristic composition
- Gitea fetch hardening with EAGER_FILTER and FULL_RESCAN options
## API Completeness
- Implemented GetCouncilComposition function
- Added GET /api/v1/councils/{id} endpoint
- Council artifacts API (POST/GET /api/v1/councils/{id}/artifacts)
- /admin/health/details endpoint with component status
- Database lookup for repository URLs (no hardcoded fallbacks)
## Observability & Performance
- Added OpenTelemetry distributed tracing with goal/pulse correlation
- Performance optimization database indexes
- Comprehensive health monitoring
- Enhanced logging and error handling
## Infrastructure
- Production-ready P2P discovery (replaces mock implementation)
- Removed unused Redis configuration
- Enhanced Docker Swarm integration
- Added migration files for performance indexes
## Code Quality
- Comprehensive input validation
- Graceful error handling and failsafe fallbacks
- Backwards compatibility maintained
- Following security best practices
🤖 Generated with [Claude Code](https://claude.ai/code)
Co-Authored-By: Claude <noreply@anthropic.com>
			
			
This commit is contained in:
		
							
								
								
									
										163
									
								
								vendor/golang.org/x/sys/unix/fstatfs_zos.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										163
									
								
								vendor/golang.org/x/sys/unix/fstatfs_zos.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,163 @@ | ||||
| // Copyright 2020 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| //go:build zos && s390x | ||||
|  | ||||
| package unix | ||||
|  | ||||
| import ( | ||||
| 	"unsafe" | ||||
| ) | ||||
|  | ||||
| // This file simulates fstatfs on z/OS using fstatvfs and w_getmntent. | ||||
|  | ||||
| func Fstatfs(fd int, stat *Statfs_t) (err error) { | ||||
| 	var stat_v Statvfs_t | ||||
| 	err = Fstatvfs(fd, &stat_v) | ||||
| 	if err == nil { | ||||
| 		// populate stat | ||||
| 		stat.Type = 0 | ||||
| 		stat.Bsize = stat_v.Bsize | ||||
| 		stat.Blocks = stat_v.Blocks | ||||
| 		stat.Bfree = stat_v.Bfree | ||||
| 		stat.Bavail = stat_v.Bavail | ||||
| 		stat.Files = stat_v.Files | ||||
| 		stat.Ffree = stat_v.Ffree | ||||
| 		stat.Fsid = stat_v.Fsid | ||||
| 		stat.Namelen = stat_v.Namemax | ||||
| 		stat.Frsize = stat_v.Frsize | ||||
| 		stat.Flags = stat_v.Flag | ||||
| 		for passn := 0; passn < 5; passn++ { | ||||
| 			switch passn { | ||||
| 			case 0: | ||||
| 				err = tryGetmntent64(stat) | ||||
| 				break | ||||
| 			case 1: | ||||
| 				err = tryGetmntent128(stat) | ||||
| 				break | ||||
| 			case 2: | ||||
| 				err = tryGetmntent256(stat) | ||||
| 				break | ||||
| 			case 3: | ||||
| 				err = tryGetmntent512(stat) | ||||
| 				break | ||||
| 			case 4: | ||||
| 				err = tryGetmntent1024(stat) | ||||
| 				break | ||||
| 			default: | ||||
| 				break | ||||
| 			} | ||||
| 			//proceed to return if: err is nil (found), err is nonnil but not ERANGE (another error occurred) | ||||
| 			if err == nil || err != nil && err != ERANGE { | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func tryGetmntent64(stat *Statfs_t) (err error) { | ||||
| 	var mnt_ent_buffer struct { | ||||
| 		header       W_Mnth | ||||
| 		filesys_info [64]W_Mntent | ||||
| 	} | ||||
| 	var buffer_size int = int(unsafe.Sizeof(mnt_ent_buffer)) | ||||
| 	fs_count, err := W_Getmntent((*byte)(unsafe.Pointer(&mnt_ent_buffer)), buffer_size) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = ERANGE //return ERANGE if no match is found in this batch | ||||
| 	for i := 0; i < fs_count; i++ { | ||||
| 		if stat.Fsid == uint64(mnt_ent_buffer.filesys_info[i].Dev) { | ||||
| 			stat.Type = uint32(mnt_ent_buffer.filesys_info[i].Fstname[0]) | ||||
| 			err = nil | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func tryGetmntent128(stat *Statfs_t) (err error) { | ||||
| 	var mnt_ent_buffer struct { | ||||
| 		header       W_Mnth | ||||
| 		filesys_info [128]W_Mntent | ||||
| 	} | ||||
| 	var buffer_size int = int(unsafe.Sizeof(mnt_ent_buffer)) | ||||
| 	fs_count, err := W_Getmntent((*byte)(unsafe.Pointer(&mnt_ent_buffer)), buffer_size) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = ERANGE //return ERANGE if no match is found in this batch | ||||
| 	for i := 0; i < fs_count; i++ { | ||||
| 		if stat.Fsid == uint64(mnt_ent_buffer.filesys_info[i].Dev) { | ||||
| 			stat.Type = uint32(mnt_ent_buffer.filesys_info[i].Fstname[0]) | ||||
| 			err = nil | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func tryGetmntent256(stat *Statfs_t) (err error) { | ||||
| 	var mnt_ent_buffer struct { | ||||
| 		header       W_Mnth | ||||
| 		filesys_info [256]W_Mntent | ||||
| 	} | ||||
| 	var buffer_size int = int(unsafe.Sizeof(mnt_ent_buffer)) | ||||
| 	fs_count, err := W_Getmntent((*byte)(unsafe.Pointer(&mnt_ent_buffer)), buffer_size) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = ERANGE //return ERANGE if no match is found in this batch | ||||
| 	for i := 0; i < fs_count; i++ { | ||||
| 		if stat.Fsid == uint64(mnt_ent_buffer.filesys_info[i].Dev) { | ||||
| 			stat.Type = uint32(mnt_ent_buffer.filesys_info[i].Fstname[0]) | ||||
| 			err = nil | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func tryGetmntent512(stat *Statfs_t) (err error) { | ||||
| 	var mnt_ent_buffer struct { | ||||
| 		header       W_Mnth | ||||
| 		filesys_info [512]W_Mntent | ||||
| 	} | ||||
| 	var buffer_size int = int(unsafe.Sizeof(mnt_ent_buffer)) | ||||
| 	fs_count, err := W_Getmntent((*byte)(unsafe.Pointer(&mnt_ent_buffer)), buffer_size) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = ERANGE //return ERANGE if no match is found in this batch | ||||
| 	for i := 0; i < fs_count; i++ { | ||||
| 		if stat.Fsid == uint64(mnt_ent_buffer.filesys_info[i].Dev) { | ||||
| 			stat.Type = uint32(mnt_ent_buffer.filesys_info[i].Fstname[0]) | ||||
| 			err = nil | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func tryGetmntent1024(stat *Statfs_t) (err error) { | ||||
| 	var mnt_ent_buffer struct { | ||||
| 		header       W_Mnth | ||||
| 		filesys_info [1024]W_Mntent | ||||
| 	} | ||||
| 	var buffer_size int = int(unsafe.Sizeof(mnt_ent_buffer)) | ||||
| 	fs_count, err := W_Getmntent((*byte)(unsafe.Pointer(&mnt_ent_buffer)), buffer_size) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	err = ERANGE //return ERANGE if no match is found in this batch | ||||
| 	for i := 0; i < fs_count; i++ { | ||||
| 		if stat.Fsid == uint64(mnt_ent_buffer.filesys_info[i].Dev) { | ||||
| 			stat.Type = uint32(mnt_ent_buffer.filesys_info[i].Fstname[0]) | ||||
| 			err = nil | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
		Reference in New Issue
	
	Block a user
	 Claude Code
					Claude Code