/* * Copyright (c) 2004 Apple Computer, Inc. All rights reserved. * * @APPLE_LICENSE_HEADER_START@ * * Copyright (c) 1999-2003 Apple Computer, Inc. All Rights Reserved. * * This file contains Original Code and/or Modifications of Original Code * as defined in and that are subject to the Apple Public Source License * Version 2.0 (the 'License'). You may not use this file except in * compliance with the License. Please obtain a copy of the License at * http://www.opensource.apple.com/apsl/ and read it before using this * file. * * The Original Code and all software distributed under the License are * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. * Please see the License for the specific language governing rights and * limitations under the License. * * @APPLE_LICENSE_HEADER_END@ Change History (most recent first): $Log: BrowserApp.java,v $ Revision 1.3 2004/05/26 01:41:58 cheshire Pass proper flags to DNSSD.enumerateDomains Revision 1.2 2004/04/30 21:53:34 rpantos Change line endings for CVS. Revision 1.1 2004/04/30 16:29:35 rpantos First checked in. BrowserApp demonstrates how to use DNSSD to browse for and resolve services. To do: - display resolved TXTRecord */ import java.awt.*; import java.awt.event.*; import java.util.*; import java.text.*; import javax.swing.*; import javax.swing.event.*; import com.apple.dnssd.*; class BrowserApp implements ListSelectionListener, ResolveListener { static BrowserApp app; JFrame frame; DomainListModel domainList; BrowserListModel servicesList, serviceList; JList domainPane, servicesPane, servicePane; DNSSDService servicesBrowser, serviceBrowser, domainBrowser; JLabel hostLabel, portLabel; public BrowserApp() { frame = new JFrame("DNS-SD Service Browser"); frame.addWindowListener(new WindowAdapter() { public void windowClosing(WindowEvent e) {System.exit(0);} }); domainList = new DomainListModel(); servicesList = new ServicesBrowserListModel(); serviceList = new BrowserListModel(); try { domainBrowser = DNSSD.enumerateDomains( DNSSD.BROWSE_DOMAINS, 0, domainList); servicesBrowser = DNSSD.browse( 0, 0, "_services._mdns._udp.", "", servicesList); serviceBrowser = null; } catch ( Exception ex) { terminateWithException( ex); } this.setupSubPanes( frame.getContentPane()); frame.pack(); frame.setVisible(true); } protected void setupSubPanes( Container parent) { parent.setLayout( new BoxLayout( parent, BoxLayout.Y_AXIS)); JPanel browserRow = new JPanel(); browserRow.setLayout( new BoxLayout( browserRow, BoxLayout.X_AXIS)); domainPane = new JList( domainList); domainPane.addListSelectionListener( this); JScrollPane domainScroller = new JScrollPane( domainPane, JScrollPane.VERTICAL_SCROLLBAR_ALWAYS, JScrollPane.HORIZONTAL_SCROLLBAR_ALWAYS); browserRow.add( domainScroller); servicesPane = new JList( servicesList); servicesPane.addListSelectionListener( this); JScrollPane servicesScroller = new JScrollPane( servicesPane, JScrollPane.VERTICAL_SCROLLBAR_ALWAYS, JScrollPane.HORIZONTAL_SCROLLBAR_ALWAYS); browserRow.add( servicesScroller); servicePane = new JList( serviceList); servicePane.addListSelectionListener( this); JScrollPane serviceScroller = new JScrollPane( servicePane, JScrollPane.VERTICAL_SCROLLBAR_ALWAYS, JScrollPane.HORIZONTAL_SCROLLBAR_ALWAYS); browserRow.add( serviceScroller); /* JPanel buttonRow = new JPanel(); buttonRow.setLayout( new BoxLayout( buttonRow, BoxLayout.X_AXIS)); buttonRow.add( Box.createHorizontalGlue()); JButton connectButton = new JButton( "Don't Connect"); buttonRow.add( connectButton); buttonRow.add( Box.createRigidArea( new Dimension( 16, 0))); */ JPanel labelRow = new JPanel(); labelRow.setLayout( new BoxLayout( labelRow, BoxLayout.X_AXIS)); labelRow.add( new JLabel( " Host: ")); hostLabel = new JLabel(); labelRow.add( hostLabel); labelRow.add( Box.createRigidArea( new Dimension( 32, 0))); labelRow.add( new JLabel( "Port: ")); portLabel = new JLabel(); labelRow.add( portLabel); labelRow.add( Box.createHorizontalGlue()); parent.add( browserRow); parent.add( Box.createRigidArea( new Dimension( 0, 8))); parent.add( labelRow); // parent.add( buttonRow); parent.add( Box.createRigidArea( new Dimension( 0, 16))); } public void valueChanged( ListSelectionEvent e) { try { if ( e.getSource() == domainPane && !e.getValueIsAdjusting()) { int newSel = domainPane.getSelectedIndex(); if ( -1 != newSel) { if ( serviceBrowser != null) serviceBrowser.stop(); serviceList.removeAllElements(); servicesBrowser = DNSSD.browse( 0, 0, "_services._mdns._udp.", "", servicesList); } } else if ( e.getSource() == servicesPane && !e.getValueIsAdjusting()) { int newSel = servicesPane.getSelectedIndex(); if ( serviceBrowser != null) serviceBrowser.stop(); serviceList.removeAllElements(); if ( -1 != newSel) serviceBrowser = DNSSD.browse( 0, 0, servicesList.getNthRegType( newSel), "", serviceList); } else if ( e.getSource() == servicePane && !e.getValueIsAdjusting()) { int newSel = servicePane.getSelectedIndex(); hostLabel.setText( ""); portLabel.setText( ""); if ( -1 != newSel) { DNSSD.resolve( 0, serviceList.getNthInterface( newSel), serviceList.getNthServiceName( newSel), serviceList.getNthRegType( newSel), serviceList.getNthDomain( newSel), new SwingResolveListener( this)); } } } catch ( Exception ex) { terminateWithException( ex); } } public void serviceResolved( DNSSDService resolver, int flags, int ifIndex, String fullName, String hostName, int port, TXTRecord txtRecord) { hostLabel.setText( hostName); portLabel.setText( String.valueOf( port)); } public void operationFailed( DNSSDService service, int errorCode) { // handle failure here } protected static void terminateWithException( Exception e) { e.printStackTrace(); System.exit( -1); } public static void main(String s[]) { app = new BrowserApp(); } } class BrowserListModel extends DefaultListModel implements BrowseListener, Runnable { public BrowserListModel() { addCache = new Vector(); removeCache = new Vector(); } /* The Browser invokes this callback when a service is discovered. */ public void serviceFound( DNSSDService browser, int flags, int ifIndex, String serviceName, String regType, String domain) { addCache.add( new BrowserListElem( serviceName, domain, regType, ifIndex)); if ( ( flags & DNSSD.MORE_COMING) == 0) this.scheduleOnEventThread(); } public void serviceLost( DNSSDService browser, int flags, int ifIndex, String serviceName, String regType, String domain) { removeCache.add( serviceName); if ( ( flags & DNSSD.MORE_COMING) == 0) this.scheduleOnEventThread(); } public void run() { while ( removeCache.size() > 0) { String serviceName = (String) removeCache.remove( removeCache.size() - 1); int matchInd = this.findMatching( serviceName); // probably doesn't handle near-duplicates well. if ( matchInd != -1) this.removeElementAt( matchInd); } while ( addCache.size() > 0) { BrowserListElem elem = (BrowserListElem) addCache.remove( addCache.size() - 1); if ( -1 == this.findMatching( elem.fServiceName)) // probably doesn't handle near-duplicates well. this.addInSortOrder( elem); } } public void operationFailed( DNSSDService service, int errorCode) { // handle failure here } /* The list contains BrowserListElem's */ class BrowserListElem { public BrowserListElem( String serviceName, String domain, String type, int ifIndex) { fServiceName = serviceName; fDomain = domain; fType = type; fInt = ifIndex; } public String toString() { return fServiceName; } public String fServiceName, fDomain, fType; public int fInt; } public String getNthServiceName( int n) { BrowserListElem sel = (BrowserListElem) this.get( n); return sel.fServiceName; } public String getNthRegType( int n) { BrowserListElem sel = (BrowserListElem) this.get( n); return sel.fType; } public String getNthDomain( int n) { BrowserListElem sel = (BrowserListElem) this.get( n); return sel.fDomain; } public int getNthInterface( int n) { BrowserListElem sel = (BrowserListElem) this.get( n); return sel.fInt; } protected void addInSortOrder( Object obj) { int i; for ( i = 0; i < this.size(); i++) if ( sCollator.compare( obj.toString(), this.getElementAt( i).toString()) < 0) break; this.add( i, obj); } protected int findMatching( String match) { for ( int i = 0; i < this.size(); i++) if ( match.equals( this.getElementAt( i).toString())) return i; return -1; } protected void scheduleOnEventThread() { try { SwingUtilities.invokeAndWait( this); } catch ( Exception e) { e.printStackTrace(); } } protected Vector removeCache; // list of serviceNames to remove protected Vector addCache; // list of BrowserListElem's to add protected static Collator sCollator; static // Initialize our static variables { sCollator = Collator.getInstance(); sCollator.setStrength( Collator.PRIMARY); } } class ServicesBrowserListModel extends BrowserListModel { /* The Browser invokes this callback when a service is discovered. */ public void serviceFound( DNSSDService browser, int flags, int ifIndex, String serviceName, String regType, String domain) // Overridden to stuff serviceName into regType and make serviceName human-readable. { regType = serviceName + ( regType.startsWith( "_udp.") ? "._udp." : "._tcp."); super.serviceFound( browser, flags, ifIndex, this.mapTypeToName( serviceName), regType, domain); } public void serviceLost( DNSSDService browser, int flags, int ifIndex, String serviceName, String regType, String domain) // Overridden to make serviceName human-readable. { super.serviceLost( browser, flags, ifIndex, this.mapTypeToName( serviceName), regType, domain); } protected String mapTypeToName( String type) // Convert a registration type into a human-readable string. Returns original string on no-match. { final String[] namedServices = { "_afpovertcp", "Apple File Sharing", "_http", "World Wide Web servers", "_daap", "Digital Audio Access", "_apple-sasl", "Apple Password Servers", "_distcc", "Distributed Compiler nodes", "_finger", "Finger servers", "_ichat", "iChat clients", "_presence", "iChat AV clients", "_ssh", "SSH servers", "_telnet", "Telnet servers", "_workstation", "Macintosh Manager clients", "_bootps", "BootP servers", "_xserveraid", "XServe RAID devices", "_eppc", "Remote AppleEvents", "_ftp", "FTP services", "_tftp", "TFTP services" }; for ( int i = 0; i < namedServices.length; i+=2) if ( namedServices[i].equals( type)) return namedServices[i + 1]; return type; } } class DomainListModel extends DefaultListModel implements DomainListener { /* Called when a domain is discovered. */ public void domainFound( DNSSDService domainEnum, int flags, int ifIndex, String domain) { if ( !this.contains( domain)) this.addElement( domain); } public void domainLost( DNSSDService domainEnum, int flags, int ifIndex, String domain) { if ( this.contains( domain)) this.removeElement( domain); } public void operationFailed( DNSSDService service, int errorCode) { // handle failure here } }