Apache Tuscany > Home > SCA Overview > SCA Java > Getting Started with Tuscany Release 1.0 | User List | Dev List | Issue Tracker |
Ready, Set, Go - Getting started with TuscanyInstall the Tuscany DistributionThe first thing you do is to create a folder on you disk into which you will download the TUSCANY distribution. Next you download the latest release distribution. Launch your browser and enter one of the
Setup Eclipse for TuscanyStart Eclipse and create a User Library to contain the TUSCANY runtime jar's as well as their From the menu bar select Window and then Preferences... . The Preferences dialog will appear, The user library created is empty, select the Add JARs... pushbutton on the right to add all the Since some of you maybe interested in debugging also the Tuscany runtime code we will attach Select the Edit... pushbutton on the right and in the Edit dialog use the External File... pushbutton Select OK to complete this and the Preferences dialog, and you are done with the Tuscany setup Create your 1st Composite Service ApplicationThe following shows the composition diagram for the composite service application you are about The composite service application you will create is a composition of four services. The composed Create a Java ProjectIn this step you create a Java Project in Eclipse to hold the composite service application. Construct ServicesFirst you create two package folders into which later in this step you place service implementations. Next you enter "services" as the package Name, and press the Finish button to complete the CatalogIn this step you create the Catalog service interface and implementation. package services; import org.osoa.sca.annotations.Remotable; @Remotable public interface Catalog { String[] get(); } Select the "services" package again. Select the New Java Class button . In the dialog enter The Java editor will open on the new created Java class. Replace the content of the editor by package services; import java.util.ArrayList; import java.util.List; import org.osoa.sca.annotations.Init; import org.osoa.sca.annotations.Property; import org.osoa.sca.annotations.Reference; public class CatalogImpl implements Catalog { @Property public String currencyCode = "USD"; @Reference public CurrencyConverter currencyConverter; private List<String> catalog = new ArrayList<String>(); @Init public void init() { String currencySymbol = currencyConverter.getCurrencySymbol(currencyCode); catalog.add("Apple - " + currencySymbol + currencyConverter.getConversion("USD", currencyCode, 2.99f)); catalog.add("Orange - " + currencySymbol + currencyConverter.getConversion("USD", currencyCode, 3.55f)); catalog.add("Pear - " + currencySymbol + currencyConverter.getConversion("USD", currencyCode, 1.55f)); } public String[] get() { String[] catalogArray = new String[catalog.size()]; catalog.toArray(catalogArray); return catalogArray; } } After completing these steps the content of the "store" project will look as follows. CurrencyConverterIn this step you create the CurrencyConverter service interface and implementation. package services; import org.osoa.sca.annotations.Remotable; @Remotable public interface CurrencyConverter { public float getConversion(String fromCurrenycCode, String toCurrencyCode, float amount); public String getCurrencySymbol(String currencyCode); } Next create a Java class in the "services" package named "CurrencyConverterImpl" and copy-paste package services; public class CurrencyConverterImpl implements CurrencyConverter { public float getConversion(String fromCurrencyCode, String toCurrencyCode, float amount) { if (toCurrencyCode.equals("USD")) return amount; else if (toCurrencyCode.equals("EUR")) return amount*0.7256f; return 0; } public String getCurrencySymbol(String currencyCode) { if (currencyCode.equals("USD")) return "$"; else if (currencyCode.equals("EUR")) return "€"; return "?"; } } After completing these steps the content of the "store" project will look as follows. ShoppingCartIn this step you create the ShoppingCart service implementation. You follow the same steps that you learned previously to create the implementation. Create a Java class in the "services" package named "ShoppingCartImpl" and copy-paste the package services; import java.util.Date; import java.util.HashMap; import java.util.Map; import java.util.UUID; import org.apache.tuscany.sca.binding.feed.collection.Collection; import org.apache.tuscany.sca.binding.feed.collection.NotFoundException; import com.sun.syndication.feed.atom.Content; import com.sun.syndication.feed.atom.Entry; import com.sun.syndication.feed.atom.Feed; import com.sun.syndication.feed.atom.Link; public class ShoppingCartImpl implements Collection { // needs to change to instance var once conversation scope works private static Map<String, Entry> cart = new HashMap<String, Entry>(); public Feed getFeed() { Feed feed = new Feed(); feed.setTitle("shopping cart"); Content subtitle = new Content(); subtitle.setValue("Total : " + getTotal()); feed.setSubtitle(subtitle); feed.getEntries().addAll(cart.values()); return feed; } public Entry get(String id) throws NotFoundException { return cart.get(id); } public Entry post(Entry entry) { String id = "cart-" + UUID.randomUUID().toString(); entry.setId(id); Link link = new Link(); link.setRel("edit"); link.setHref("" + id); entry.getOtherLinks().add(link); link = new Link(); link.setRel("alternate"); link.setHref("" + id); entry.getAlternateLinks().add(link); entry.setCreated(new Date()); cart.put(id, entry); return entry; } public Entry put(String id, Entry entry) throws NotFoundException { entry.setUpdated(new Date()); cart.put(id, entry); return entry; } public void delete(String id) throws NotFoundException { if (id.equals("")) cart.clear(); else cart.remove(id); } private String getTotal() { float total = 0; String symbol = ""; if (!cart.isEmpty()) { Entry entry = cart.values().iterator().next(); String item = ((Content)entry.getContents().get(0)).getValue(); symbol = item.substring(item.indexOf("-")+2, item.indexOf("-")+3); } for (Entry entry : cart.values()) { String item = ((Content)entry.getContents().get(0)).getValue(); total += Float.valueOf(item.substring(item.indexOf("-")+3)); } return symbol + String.valueOf(total); } } Note: Since the Tuscany conversational support is not ready yet the cart is realized through a hack. After completing these steps the content of the "store" project will look as follows.
StoreIn this step you create the user facing Store service that will run in a Web browser and provide the Select the "ufservices" package. Right click to get the context menu, select New, and then File. In The Text editor will open on the new created html file. Replace the content of the editor by copy-paste <html> <head> <title>Store</title> <script type="text/javascript" src="binding-atom.js"></script> <script type="text/javascript" src="binding-jsonrpc.js"></script> <script language="JavaScript"> //Reference catalog = (new JSONRpcClient("../Catalog")).Catalog; //Reference shoppingCart = new AtomClient("../ShoppingCart"); function catalog_getResponse(items) { var catalog = ""; for (var i=0; i<items.length; i++) catalog += '<input name="items" type="checkbox" value="' + items[i] + '">' + items[i]+ ' <br>'; document.getElementById('catalog').innerHTML=catalog; } function shoppingCart_getResponse(feed) { if (feed != null) { var entries = feed.getElementsByTagName("entry"); var list = ""; for (var i=0; i<entries.length; i++) { var item = entries[i].getElementsByTagName("content")[0].firstChild.nodeValue; list += item + ' <br>'; } document.getElementById("shoppingCart").innerHTML = list; if (list != "") document.getElementById('total').innerHTML = feed.getElementsByTagName("subtitle")[0].firstChild.nodeValue; } } function shoppingCart_postResponse(entry) { shoppingCart.get("", shoppingCart_getResponse); } function addToCart() { var items = document.catalogForm.items; var j = 0; for (var i=0; i<items.length; i++) if (items[i].checked) { var entry = '<entry xmlns="http://www.w3.org/2005/Atom">' + '<title>cart-item</title>' + '<content type="text">'+items[i].value+'</content>' + '</entry>'; shoppingCart.post(entry, shoppingCart_postResponse); items[i].checked = false; } } function checkoutCart() { document.getElementById('store').innerHTML='<h2>' + 'Thanks for Shopping With Us!</h2>'+ '<h2>Your Order</h2>'+ '<form name="orderForm" action="/ufs/store.html">'+ document.getElementById('shoppingCart').innerHTML+ '<br>'+ document.getElementById('total').innerHTML+ '<br>'+ '<br>'+ '<input type="submit" value="Continue Shopping">'+ '</form>'; shoppingCart.delete("", null); } function deleteCart() { shoppingCart.delete("", null); document.getElementById('shoppingCart').innerHTML = ""; document.getElementById('total').innerHTML = ""; } window.onload = function() { catalog.get(catalog_getResponse); shoppingCart.get("", shoppingCart_getResponse); } </script> <link rel="stylesheet" type="text/css" href="style.css" /> </head> <body> <h1>Store</h1> <div id="store"> <h2>Catalog</h2> <form name="catalogForm"> <div id="catalog" ></div> <br> <input type="button" onClick="addToCart()" value="Add to Cart"> </form> <br> <h2>Your Shopping Cart</h2> <form name="shoppingCartForm"> <div id="shoppingCart"></div> <br> <div id="total"></div> <br> <input type="button" onClick="checkoutCart()" value="Checkout"> <input type="button" onClick="deleteCart()" value="Empty"> <a href="../ShoppingCart/"> <img src="http://projects.apache.org/images/blue-icon.png" border="0"> </a> </form> </div> </body> </html> Next select the "ufservices" package again. Right click to get the context menu, select New, and The Text editor will open on the new created javascript file. Replace the content of the editor by Next select the "ufservices" package again. Right click to get the context menu, select New, and The Text editor will open on the new created javascript file. Replace the content of the editor by function AtomClient(uri) { this.uri=uri; this.get = function(id, responseFunction) { var xhr = this.createXMLHttpRequest(); xhr.onreadystatechange = function() { if (xhr.readyState == 4) { if (xhr.status == 200) { if (responseFunction != null) responseFunction(xhr.responseXML); } else { alert("get - Error getting data from the server"); } } } xhr.open("GET", uri + id, true); xhr.send(null); } this.post = function (entry, responseFunction) { var xhr = this.createXMLHttpRequest(); xhr.onreadystatechange = function() { if (xhr.readyState == 4) { if (xhr.status == 201) { if (responseFunction != null) responseFunction(xhr.responseXML); } else { alert("post - Error getting data from the server"); } } } xhr.open("POST", uri, true); xhr.setRequestHeader("Content-Type", "application/atom+xml"); xhr.send(entry); } this.put = function (id, entry, responseFunction) { var xhr = this.createXMLHttpRequest(); xhr.onreadystatechange = function() { if (xhr.readyState == 4) { if (xhr.status == 200) { if (responseFunction != null) responseFunction(xhr.responseXML); } else { alert("put - Error getting data from the server"); } } } xhr.open("PUT", uri + id, true); xhr.setRequestHeader("Content-Type", "application/atom+xml"); xhr.send(entry); } this.delete = function (id, responseFunction) { var xhr = this.createXMLHttpRequest(); xhr.onreadystatechange = function() { if (xhr.readyState == 4) { if (xhr.status == 200) { if (responseFunction != null) responseFunction(); } else { alert("delete - Error getting data from the server"); } } } xhr.open("DELETE", uri + id, true); xhr.send(null); } this.createXMLHttpRequest = function () { try {return new XMLHttpRequest();} catch(e) {} try {return new ActiveXObject("Msxml2.XMLHTTP");} catch(e) {} try {return new ActiveXObject("Microsoft.XMLHTTP");} catch(e) {} alert("XML http request not supported"); return null; } } Note: That we have to have the bindig-jsonrpc.js, and binding-atom.js local in our project is only After completing these steps the content of the "store" project will look as follows.
Compose ServicesNow that you have all the required service implementations you compose them together to provide Select the "src" folder of the "store" project. Right click to get the context menu, select New, and The Text editor will open on the new created composite file. Replace the content of the editor by <?xml version="1.0" encoding="UTF-8"?> <composite xmlns="http://www.osoa.org/xmlns/sca/1.0" xmlns:t="http://tuscany.apache.org/xmlns/sca/1.0" xmlns:s="http://store" name="store"> <component name="ufs"> <t:implementation.resource location="ufservices"/> <service name="Resource"> <t:binding.http/> </service> </component> <component name="Catalog"> <implementation.java class="services.CatalogImpl"/> <property name="currencyCode">USD</property> <service name="Catalog"> <t:binding.jsonrpc/> </service> <reference name="currencyConverter" target="CurrencyConverter"/> </component> <component name="ShoppingCart"> <implementation.java class="services.ShoppingCartImpl"/> <service name="Collection"> <t:binding.atom/> </service> </component> <component name="CurrencyConverter"> <implementation.java class="services.CurrencyConverterImpl"/> </component> </composite> After completing these steps the content of the "store" project will look as follows.
Launch ServicesIn this step you create the code to launch the Tuscany runtime with the new store composite Select the "store" project and click on the New Java Package button in the toolbar to start the package creation dialog. Use the dialog to create a new package named "launch". The Java editor will open on the new created Java class. Replace the content of the editor by package launch; import org.apache.tuscany.sca.host.embedded.SCADomain; public class Launch { public static void main(String[] args) throws Exception { System.out.println("Starting ..."); SCADomain scaDomain = SCADomain.newInstance("store.composite"); System.out.println("store.composite ready for big business !!!"); System.out.println(); System.in.read(); scaDomain.close(); } } After completing these steps the content of the "store" project will look as follows.
Use ServicesIn this step you launch and use the store composite service application you created. First select the "Launch" class in the "launch" package of your "store" project. Right click to get the The Eclipse console will show the following messages.
Next Launch your Web browser and enter the following address: http://localhost:8080/ufs/store.html You get to the Store user facing service of the composite service application. You can select items from the Catalog and add them to your Shopping Cart. Note: When adding items for the first time you will be asked for userid and password by the
Since the ShoppingCart service is bound using the ATOM binding, you can also look at the
Use the browser back button to get back to the Store page. And then you can Checkout to complete your order. Explore the Samples from the Tuscany DistributionThe sample folder of the Tuscany distribution provides a rich set of samples ready for you to explore.
|