You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							151 lines
						
					
					
						
							7.8 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							151 lines
						
					
					
						
							7.8 KiB
						
					
					
				
								package app
							 | 
						|
								
							 | 
						|
								import "fmt"
							 | 
						|
								import "github.com/seaweedfs/seaweedfs/weed/admin/dash"
							 | 
						|
								
							 | 
						|
								templ Subscribers(data dash.SubscribersData) {
							 | 
						|
								    <div class="container-fluid">
							 | 
						|
								        <div class="row">
							 | 
						|
								            <div class="col-12">
							 | 
						|
								                <div class="d-flex justify-content-between align-items-center mb-4">
							 | 
						|
								                    <h1 class="h3 mb-0">Message Queue Subscribers</h1>
							 | 
						|
								                    <small class="text-muted">Last updated: {data.LastUpdated.Format("2006-01-02 15:04:05")}</small>
							 | 
						|
								                </div>
							 | 
						|
								
							 | 
						|
								                <!-- Summary Cards -->
							 | 
						|
								                <div class="row mb-4">
							 | 
						|
								                    <div class="col-md-4">
							 | 
						|
								                        <div class="card text-center">
							 | 
						|
								                            <div class="card-body">
							 | 
						|
								                                <h5 class="card-title">Total Subscribers</h5>
							 | 
						|
								                                <h3 class="text-primary">{fmt.Sprintf("%d", data.TotalSubscribers)}</h3>
							 | 
						|
								                            </div>
							 | 
						|
								                        </div>
							 | 
						|
								                    </div>
							 | 
						|
								                    <div class="col-md-4">
							 | 
						|
								                        <div class="card text-center">
							 | 
						|
								                            <div class="card-body">
							 | 
						|
								                                <h5 class="card-title">Active Subscribers</h5>
							 | 
						|
								                                <h3 class="text-success">{fmt.Sprintf("%d", data.ActiveSubscribers)}</h3>
							 | 
						|
								                            </div>
							 | 
						|
								                        </div>
							 | 
						|
								                    </div>
							 | 
						|
								                    <div class="col-md-4">
							 | 
						|
								                        <div class="card text-center">
							 | 
						|
								                            <div class="card-body">
							 | 
						|
								                                <h5 class="card-title">Inactive Subscribers</h5>
							 | 
						|
								                                <h3 class="text-warning">{fmt.Sprintf("%d", data.TotalSubscribers - data.ActiveSubscribers)}</h3>
							 | 
						|
								                            </div>
							 | 
						|
								                        </div>
							 | 
						|
								                    </div>
							 | 
						|
								                </div>
							 | 
						|
								
							 | 
						|
								                <!-- Subscribers Table -->
							 | 
						|
								                <div class="card">
							 | 
						|
								                    <div class="card-header d-flex justify-content-between align-items-center">
							 | 
						|
								                        <h5 class="mb-0">Subscribers</h5>
							 | 
						|
								                        <div>
							 | 
						|
								                            <button class="btn btn-sm btn-outline-secondary" onclick="exportSubscribersCSV()">
							 | 
						|
								                                <i class="fas fa-download me-1"></i>Export CSV
							 | 
						|
								                            </button>
							 | 
						|
								                        </div>
							 | 
						|
								                    </div>
							 | 
						|
								                    <div class="card-body">
							 | 
						|
								                        if len(data.Subscribers) == 0 {
							 | 
						|
								                            <div class="text-center py-4">
							 | 
						|
								                                <i class="fas fa-user-friends fa-3x text-muted mb-3"></i>
							 | 
						|
								                                <h5>No Subscribers Found</h5>
							 | 
						|
								                                <p class="text-muted">No message queue subscribers are currently active.</p>
							 | 
						|
								                            </div>
							 | 
						|
								                        } else {
							 | 
						|
								                            <div class="table-responsive">
							 | 
						|
								                                <table class="table table-striped" id="subscribersTable">
							 | 
						|
								                                    <thead>
							 | 
						|
								                                        <tr>
							 | 
						|
								                                            <th>Subscriber Name</th>
							 | 
						|
								                                            <th>Topic</th>
							 | 
						|
								                                            <th>Consumer Group</th>
							 | 
						|
								                                            <th>Status</th>
							 | 
						|
								                                            <th>Messages Processed</th>
							 | 
						|
								                                            <th>Last Seen</th>
							 | 
						|
								                                            <th>Created</th>
							 | 
						|
								                                        </tr>
							 | 
						|
								                                    </thead>
							 | 
						|
								                                    <tbody>
							 | 
						|
								                                        for _, subscriber := range data.Subscribers {
							 | 
						|
								                                            <tr>
							 | 
						|
								                                                <td>
							 | 
						|
								                                                    <strong>{subscriber.Name}</strong>
							 | 
						|
								                                                </td>
							 | 
						|
								                                                <td>
							 | 
						|
								                                                    <span class="badge bg-info">{subscriber.Topic}</span>
							 | 
						|
								                                                </td>
							 | 
						|
								                                                <td>{subscriber.ConsumerGroup}</td>
							 | 
						|
								                                                <td>
							 | 
						|
								                                                    if subscriber.Status == "active" {
							 | 
						|
								                                                        <span class="badge bg-success">Active</span>
							 | 
						|
								                                                    } else if subscriber.Status == "inactive" {
							 | 
						|
								                                                        <span class="badge bg-warning">Inactive</span>
							 | 
						|
								                                                    } else {
							 | 
						|
								                                                        <span class="badge bg-secondary">{subscriber.Status}</span>
							 | 
						|
								                                                    }
							 | 
						|
								                                                </td>
							 | 
						|
								                                                <td>{fmt.Sprintf("%d", subscriber.MessageCount)}</td>
							 | 
						|
								                                                <td>
							 | 
						|
								                                                    if !subscriber.LastSeen.IsZero() {
							 | 
						|
								                                                        <span class="text-muted">{subscriber.LastSeen.Format("2006-01-02 15:04:05")}</span>
							 | 
						|
								                                                    } else {
							 | 
						|
								                                                        <span class="text-muted">Never</span>
							 | 
						|
								                                                    }
							 | 
						|
								                                                </td>
							 | 
						|
								                                                <td>
							 | 
						|
								                                                    <span class="text-muted">{subscriber.CreatedAt.Format("2006-01-02 15:04:05")}</span>
							 | 
						|
								                                                </td>
							 | 
						|
								                                            </tr>
							 | 
						|
								                                        }
							 | 
						|
								                                    </tbody>
							 | 
						|
								                                </table>
							 | 
						|
								                            </div>
							 | 
						|
								                        }
							 | 
						|
								                    </div>
							 | 
						|
								                </div>
							 | 
						|
								            </div>
							 | 
						|
								        </div>
							 | 
						|
								    </div>
							 | 
						|
								
							 | 
						|
								    <script>
							 | 
						|
								        function exportSubscribersCSV() {
							 | 
						|
								            const table = document.getElementById('subscribersTable');
							 | 
						|
								            if (!table) return;
							 | 
						|
								            
							 | 
						|
								            let csv = 'Subscriber Name,Topic,Consumer Group,Status,Messages Processed,Last Seen,Created\n';
							 | 
						|
								            
							 | 
						|
								            const rows = table.querySelectorAll('tbody tr');
							 | 
						|
								            rows.forEach(row => {
							 | 
						|
								                const cells = row.querySelectorAll('td');
							 | 
						|
								                if (cells.length >= 7) {
							 | 
						|
								                    const rowData = [
							 | 
						|
								                        cells[0].querySelector('strong')?.textContent || '',
							 | 
						|
								                        cells[1].querySelector('.badge')?.textContent || '',
							 | 
						|
								                        cells[2].textContent || '',
							 | 
						|
								                        cells[3].querySelector('.badge')?.textContent || '',
							 | 
						|
								                        cells[4].textContent || '',
							 | 
						|
								                        cells[5].querySelector('span')?.textContent || '',
							 | 
						|
								                        cells[6].querySelector('span')?.textContent || ''
							 | 
						|
								                    ];
							 | 
						|
								                    csv += rowData.map(field => `"${field.replace(/"/g, '""')}"`).join(',') + '\n';
							 | 
						|
								                }
							 | 
						|
								            });
							 | 
						|
								            
							 | 
						|
								            const blob = new Blob([csv], { type: 'text/csv;charset=utf-8;' });
							 | 
						|
								            const link = document.createElement('a');
							 | 
						|
								            const url = URL.createObjectURL(blob);
							 | 
						|
								            link.setAttribute('href', url);
							 | 
						|
								            link.setAttribute('download', 'subscribers.csv');
							 | 
						|
								            link.style.visibility = 'hidden';
							 | 
						|
								            document.body.appendChild(link);
							 | 
						|
								            link.click();
							 | 
						|
								            document.body.removeChild(link);
							 | 
						|
								        }
							 | 
						|
								    </script>
							 | 
						|
								} 
							 |