डायनॉम्बीडी - ग्लोबल सेकेंडरी इंडेक्स

विभिन्न विशेषताओं के साथ विभिन्न क्वेरी प्रकारों की आवश्यकता वाले अनुप्रयोग इन विस्तृत प्रश्नों को करने में एकल या एकाधिक वैश्विक माध्यमिक अनुक्रमित का उपयोग कर सकते हैं।

For example - एक प्रणाली जो उपयोगकर्ताओं का ट्रैक रखती है, उनकी लॉगिन स्थिति और उनका समय लॉग इन होता है। पिछले उदाहरण की वृद्धि इसके डेटा पर प्रश्नों को धीमा कर देती है।

वैश्विक माध्यमिक सूचकांक एक तालिका से विशेषताओं के चयन का आयोजन करके प्रश्नों में तेजी लाते हैं। वे डेटा को सॉर्ट करने में प्राथमिक कुंजी को नियोजित करते हैं, और तालिका के समान कोई कुंजी तालिका विशेषताएँ, या कुंजी स्कीमा की आवश्यकता नहीं होती है।

सभी वैश्विक सेकेंडरी इंडेक्स में एक सॉर्ट कुंजी के विकल्प के साथ एक विभाजन कुंजी शामिल होनी चाहिए। अनुक्रमणिका कुंजी स्कीमा तालिका से भिन्न हो सकती है, और अनुक्रमणिका कुंजी विशेषताएँ किसी भी शीर्ष-स्तरीय स्ट्रिंग, संख्या, या बाइनरी तालिका विशेषताओं का उपयोग कर सकती हैं।

एक प्रक्षेपण में, आप अन्य तालिका विशेषताओं का उपयोग कर सकते हैं, हालांकि, माता-पिता की तालिकाओं से प्रश्न पुनः प्राप्त नहीं होते हैं।

गुण का अनुमान

अनुमानों में तालिका से द्वितीयक सूचकांक में कॉपी किए गए विशेषता सेट होते हैं। हमेशा तालिका विभाजन कुंजी और सॉर्ट कुंजी के साथ एक प्रोजेक्शन होता है। प्रश्नों में, अनुमान प्रक्षेपण की किसी भी विशेषता के लिए डायनेमोडी एक्सेस की अनुमति देते हैं; वे अनिवार्य रूप से अपनी तालिका के रूप में मौजूद हैं।

एक माध्यमिक सूचकांक निर्माण में, आपको प्रक्षेपण के लिए विशेषताओं को निर्दिष्ट करना होगा। DynamoDB इस कार्य को करने के लिए तीन तरीके प्रदान करता है -

  • KEYS_ONLY- सभी अनुक्रमणिका आइटम तालिका विभाजन से मिलकर बने होते हैं और प्रमुख मानों को छांटते हैं, और प्रमुख कुंजी मानों को। यह सबसे छोटा सूचकांक बनाता है।

  • INCLUDE - इसमें KEYS_ONLY विशेषताएँ और निर्दिष्ट गैर-प्रमुख विशेषताएँ शामिल हैं।

  • ALL - इसमें सभी स्रोत तालिका विशेषताएँ शामिल हैं, जो सबसे बड़ा संभव सूचकांक बनाता है।

एक ग्लोबल सेकेंडरी इंडेक्स में प्रोजेक्टिंग विशेषताओं में ट्रेडऑफ पर ध्यान दें, जो थ्रूपुट और भंडारण लागत से संबंधित है।

निम्नलिखित बिंदुओं पर विचार करें -

  • यदि आपको केवल कम विलंबता के साथ कुछ विशेषताओं तक पहुंच की आवश्यकता है, तो केवल उन प्रोजेक्ट्स की आवश्यकता है जिन्हें आप की आवश्यकता है। यह भंडारण को कम करता है और लागत लिखता है।

  • यदि कोई एप्लिकेशन अक्सर कुछ गैर-कुंजी विशेषताओं तक पहुंचता है, तो उन्हें प्रोजेक्ट करें क्योंकि स्कैन की खपत की तुलना में भंडारण की लागत कम है।

  • आप अक्सर एक्सेस की जाने वाली विशेषताओं के बड़े सेटों को प्रोजेक्ट कर सकते हैं, हालांकि, यह उच्च भंडारण लागत वहन करती है।

  • बार-बार लिखने और अपडेट के लिए KEYS_ONLY का उपयोग करें। यह आकार को नियंत्रित करता है, लेकिन फिर भी प्रश्नों पर अच्छा प्रदर्शन प्रदान करता है।

वैश्विक माध्यमिक सूचकांक प्रश्न और स्कैन

किसी इंडेक्स में एक या कई आइटम्स को एक्सेस करने के लिए आप प्रश्नों का उपयोग कर सकते हैं। आपको सूचकांक और तालिका का नाम, वांछित विशेषताएँ और शर्तें निर्दिष्ट करनी चाहिए; आरोही या अवरोही क्रम में परिणाम वापस करने के विकल्प के साथ।

आप सभी सूचकांक डेटा प्राप्त करने के लिए स्कैन का भी उपयोग कर सकते हैं। इसके लिए टेबल और इंडेक्स नाम चाहिए। विशिष्ट डेटा प्राप्त करने के लिए आप एक फ़िल्टर एक्सप्रेशन का उपयोग करते हैं।

तालिका और सूचकांक डेटा तुल्यकालन

डायनेमोडीबी स्वचालित रूप से अपनी मूल तालिका के साथ अनुक्रमित पर सिंक्रनाइज़ेशन करता है। आइटम पर प्रत्येक संशोधन संचालन अतुल्यकालिक अद्यतन का कारण बनता है, हालांकि, अनुप्रयोग सीधे अनुक्रमित करने के लिए नहीं लिखते हैं।

आपको सूचकांकों पर डायनेमोडीबी रखरखाव के प्रभाव को समझने की आवश्यकता है। एक इंडेक्स के निर्माण पर, आप मुख्य विशेषताओं और डेटा प्रकारों को निर्दिष्ट करते हैं, जिसका अर्थ है एक लेखन पर, उन डेटा प्रकारों को कुंजी स्कीमा डेटा प्रकारों से मेल खाना चाहिए।

आइटम निर्माण या विलोपन पर, अनुक्रमणिका अंततः एक सुसंगत तरीके से अपडेट होती है, हालांकि, डेटा को दूसरे के एक अंश में प्रसारित करने के लिए अपडेट होता है (जब तक कि कुछ प्रकार की सिस्टम विफलता नहीं होती है)। इस देरी के लिए आपको आवेदन करना होगा।

Throughput Considerations in Global Secondary Indexes- मल्टीपल ग्लोबल सेकेंडरी इंडेक्स थ्रूपुट को प्रभावित करते हैं। सूचकांक निर्माण के लिए क्षमता इकाई विनिर्देशों की आवश्यकता होती है, जो तालिका से अलग होती हैं, जिसके परिणामस्वरूप तालिका इकाइयों के बजाय सूचकांक क्षमता इकाइयों का संचालन होता है।

यह थ्रॉटलिंग में परिणाम कर सकता है यदि कोई क्वेरी या लिखी गई प्रावधान थ्रूपुट से अधिक है। उपयोग करके थ्रूपुट सेटिंग्स देखेंDescribeTable

Read Capacity- ग्लोबल सेकेंडरी इंडेक्स अंततः स्थिरता प्रदान करते हैं। प्रश्नों में, डायनेमोडी टेबल के लिए उपयोग किए जाने वाले प्रावधान गणनाओं को निष्पादित करता है, जिसमें आइटम आकार के बजाय सूचकांक प्रविष्टि आकार का उपयोग करने का एक अकेला अंतर होता है। क्वेरी रिटर्न की सीमा 1 एमबी रहती है, जिसमें विशेषता नाम का आकार और प्रत्येक लौटे आइटम में मान शामिल होते हैं।

क्षमता लिखें

जब लिखने का कार्य होता है, तो प्रभावित सूचकांक लेखन इकाइयों का उपभोग करता है। तालिका के माध्यम से लिखी जाने वाली कैपेसिटी यूनिट्स और इंडेक्स अपडेट्स में खपत यूनिट्स का योग लिखें। एक सफल लेखन ऑपरेशन के लिए पर्याप्त क्षमता की आवश्यकता होती है, या इसका परिणाम थ्रॉटलिंग में होता है।

लिखने की लागत भी कुछ कारकों पर निर्भर रहती है, जिनमें से कुछ इस प्रकार हैं -

  • अनुक्रमित विशेषताओं या आइटम अपडेट को परिभाषित करने वाले नए आइटम अपरिभाषित अनुक्रमित विशेषताओं को परिभाषित करते हुए, आइटम को इंडेक्स में जोड़ने के लिए एकल लेखन ऑपरेशन का उपयोग करते हैं।

  • अनुक्रमित कुंजी विशेषता मान को बदलने वाले अपडेट एक आइटम को हटाने के लिए दो लिखते हैं और एक नया लिखते हैं।

  • अनुक्रमणित विशेषता का एक तालिका लेखन ट्रिगरिंग विलोपन सूचकांक में पुराने आइटम प्रक्षेपण को मिटाने के लिए एकल लेखन का उपयोग करता है।

  • सूचकांक में अनुपस्थित आइटम एक अद्यतन कार्रवाई के उपयोग से पहले और बाद में कोई लिखता नहीं है।

  • अनुक्रमणिका कुंजी स्कीमा में केवल अनुमानित विशेषता मान को बदलने वाले अपडेट, और अनुक्रमित कुंजी विशेषता मान नहीं, इंडेक्स में अनुमानित विशेषताओं के अद्यतन को लिखने के लिए एक लिखने का उपयोग करें।

ये सभी कारक 1KB से कम या इसके बराबर के आइटम आकार को मानते हैं।

वैश्विक माध्यमिक सूचकांक भंडारण

आइटम लिखने पर, DynamoDB स्वचालित रूप से उन विशेषताओं के सही सेट की प्रतिलिपि बना देता है, जहाँ विशेषताएँ मौजूद होनी चाहिए। यह आपके खाते को तालिका आइटम संग्रहण और विशेषता संग्रहण के लिए चार्ज करके प्रभावित करता है। इन राशियों के योग से परिणाम का उपयोग किया जाता है -

  • तालिका प्राथमिक कुंजी का बाइट आकार
  • सूचकांक कुंजी विशेषता का बाइट आकार
  • अनुमानित विशेषताओं का बाइट आकार
  • प्रति इंडेक्स आइटम पर 100 बाइट-ओवरहेड

आप औसत आइटम आकार का आकलन करने और वैश्विक माध्यमिक सूचकांक कुंजी विशेषताओं के साथ तालिका आइटम की मात्रा से गुणा करके भंडारण की जरूरतों का अनुमान लगा सकते हैं।

डायनेमोबीडी एक तालिका आइटम के लिए आइटम डेटा नहीं लिखता है जिसमें एक अनिर्धारित विशेषता के साथ एक इंडेक्स विभाजन या सॉर्ट कुंजी के रूप में परिभाषित किया गया है।

ग्लोबल सेकेंडरी इंडेक्स क्रूड

का उपयोग करके वैश्विक माध्यमिक अनुक्रमित के साथ एक तालिका बनाएँ CreateTable ऑपरेशन के साथ जोड़ा गया GlobalSecondaryIndexesपैरामीटर। आपको इंडेक्स विभाजन कुंजी के रूप में सेवा करने के लिए या इंडेक्स सॉर्ट कुंजी के लिए किसी अन्य का उपयोग करने के लिए एक विशेषता निर्दिष्ट करनी होगी। सभी सूचकांक कुंजी विशेषताएँ स्ट्रिंग, संख्या या बाइनरी स्केलर होनी चाहिए। आपको थ्रूपुट सेटिंग्स भी प्रदान करनी चाहिए, जिसमें शामिल हैReadCapacityUnits तथा WriteCapacityUnits

उपयोग UpdateTable GlobalSecondaryIndexes पैरामीटर का उपयोग करके एक बार फिर से वैश्विक सेकेंडरी इंडेक्स को मौजूदा तालिकाओं में जोड़ना।

इस ऑपरेशन में, आपको निम्नलिखित जानकारी प्रदान करनी होगी -

  • सूचकांक नाम
  • मुख्य स्कीमा
  • अनुमानित विशेषताएँ
  • थ्रूपुट सेटिंग्स

एक वैश्विक माध्यमिक सूचकांक जोड़कर, आइटम की मात्रा, अनुमानित गुण मात्रा, लिखने की क्षमता और लेखन गतिविधि के कारण बड़ी तालिकाओं के साथ पर्याप्त समय लग सकता है। उपयोगCloudWatch प्रक्रिया पर नजर रखने के लिए मैट्रिक्स।

उपयोग DescribeTableएक वैश्विक माध्यमिक सूचकांक के लिए स्थिति की जानकारी प्राप्त करना। यह चार में से एक लौटाता हैIndexStatus GlobalSecondaryIndexes के लिए -

  • CREATING - यह सूचकांक के निर्माण चरण, और इसकी अनुपलब्धता को इंगित करता है।

  • ACTIVE - यह उपयोग के लिए सूचकांक की तत्परता को इंगित करता है।

  • UPDATING - यह थ्रूपुट सेटिंग्स की अद्यतन स्थिति को इंगित करता है।

  • DELETING - यह सूचकांक की डिलीट स्थिति और उपयोग के लिए इसकी स्थायी अनुपलब्धता को इंगित करता है।

अद्यतन ग्लोबल सेकंड इंडेक्स लोडिंग / बैकफ़िलिंग चरण (डायनेमोबीडी लेखन विशेषताओं को एक इंडेक्स और जोड़े गए / हटाए गए / अपडेट किए गए आइटम को ट्रैक करने के दौरान) थ्रूपुट सेटिंग्स का प्रावधान करता है। उपयोगUpdateTable इस ऑपरेशन को करने के लिए।

आपको याद रखना चाहिए कि आप बैकफ़िलिंग चरण के दौरान अन्य सूचकांकों को जोड़ / हटा नहीं सकते हैं।

ग्लोबल सेकेंडरी इंडेक्स को हटाने के लिए UpdateTable का उपयोग करें। यह प्रति ऑपरेशन केवल एक इंडेक्स को हटाने की अनुमति देता है, हालांकि, आप कई ऑपरेशनों को समवर्ती रूप से पांच तक चला सकते हैं। हटाने की प्रक्रिया पैरेंट टेबल की पढ़ने / लिखने की गतिविधियों को प्रभावित नहीं करती है, लेकिन ऑपरेशन पूरा होने तक आप अन्य सूचकांकों को जोड़ / हटा नहीं सकते हैं।

ग्लोबल सेकेंडरी इंडेक्स के साथ काम करने के लिए जावा का उपयोग करना

CreateTable के माध्यम से एक इंडेक्स के साथ एक तालिका बनाएं। बस एक DynamoDB वर्ग उदाहरण बनाएँ, aCreateTableRequest अनुरोध जानकारी के लिए वर्ग उदाहरण, और CreateTable विधि के लिए अनुरोध ऑब्जेक्ट पास करें।

निम्नलिखित कार्यक्रम एक छोटा उदाहरण है -

DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient ( 
   new ProfileCredentialsProvider()));
   
// Attributes 
ArrayList<AttributeDefinition> attributeDefinitions = new 
   ArrayList<AttributeDefinition>();  
attributeDefinitions.add(new AttributeDefinition() 
   .withAttributeName("City") 
   .withAttributeType("S"));
   
attributeDefinitions.add(new AttributeDefinition() 
   .withAttributeName("Date") 
   .withAttributeType("S"));
   
attributeDefinitions.add(new AttributeDefinition() 
   .withAttributeName("Wind") 
   .withAttributeType("N"));
   
// Key schema of the table 
ArrayList<KeySchemaElement> tableKeySchema = new ArrayList<KeySchemaElement>(); 
tableKeySchema.add(new KeySchemaElement()
   .withAttributeName("City") 
   .withKeyType(KeyType.HASH));              //Partition key
   
tableKeySchema.add(new KeySchemaElement() 
   .withAttributeName("Date") 
   .withKeyType(KeyType.RANGE));             //Sort key
   
// Wind index 
GlobalSecondaryIndex windIndex = new GlobalSecondaryIndex() 
   .withIndexName("WindIndex") 
   .withProvisionedThroughput(new ProvisionedThroughput() 
   .withReadCapacityUnits((long) 10) 
   .withWriteCapacityUnits((long) 1)) 
   .withProjection(new Projection().withProjectionType(ProjectionType.ALL));
   
ArrayList<KeySchemaElement> indexKeySchema = new ArrayList<KeySchemaElement>(); 
indexKeySchema.add(new KeySchemaElement() 
   .withAttributeName("Date") 
   .withKeyType(KeyType.HASH));              //Partition key
   
indexKeySchema.add(new KeySchemaElement() 
   .withAttributeName("Wind") 
   .withKeyType(KeyType.RANGE));             //Sort key
   
windIndex.setKeySchema(indexKeySchema);  
CreateTableRequest createTableRequest = new CreateTableRequest() 
   .withTableName("ClimateInfo") 
   .withProvisionedThroughput(new ProvisionedThroughput() 
   .withReadCapacityUnits((long) 5) 
   .withWriteCapacityUnits((long) 1))
   .withAttributeDefinitions(attributeDefinitions) 
   .withKeySchema(tableKeySchema) 
   .withGlobalSecondaryIndexes(windIndex); 
Table table = dynamoDB.createTable(createTableRequest); 
System.out.println(table.getDescription());

सूचकांक जानकारी को पुनः प्राप्त करें DescribeTable। सबसे पहले, DynamoDB वर्ग उदाहरण बनाएं। फिर एक सूचकांक को लक्षित करने के लिए एक टेबल क्लास का उदाहरण बनाएं। अंत में, तालिका का वर्णन विधि में पास करें।

यहाँ एक छोटा उदाहरण है -

DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient ( 
   new ProfileCredentialsProvider()));
   
Table table = dynamoDB.getTable("ClimateInfo"); 
TableDescription tableDesc = table.describe();  
Iterator<GlobalSecondaryIndexDescription> gsiIter = 
   tableDesc.getGlobalSecondaryIndexes().iterator(); 

while (gsiIter.hasNext()) { 
   GlobalSecondaryIndexDescription gsiDesc = gsiIter.next(); 
   System.out.println("Index data " + gsiDesc.getIndexName() + ":");  
   Iterator<KeySchemaElement> kse7Iter = gsiDesc.getKeySchema().iterator(); 
   
   while (kseIter.hasNext()) { 
      KeySchemaElement kse = kseIter.next(); 
      System.out.printf("\t%s: %s\n", kse.getAttributeName(), kse.getKeyType()); 
   }
   Projection projection = gsiDesc.getProjection(); 
   System.out.println("\tProjection type: " + projection.getProjectionType()); 
   
   if (projection.getProjectionType().toString().equals("INCLUDE")) { 
      System.out.println("\t\tNon-key projected attributes: " 
         + projection.getNonKeyAttributes()); 
   } 
}

तालिका क्वेरी के साथ अनुक्रमणिका क्वेरी करने के लिए क्वेरी का उपयोग करें। बस डायनामो बी बी श्रेणी उदाहरण, लक्ष्य सूचकांक के लिए एक तालिका वर्ग उदाहरण, विशिष्ट सूचकांक के लिए एक सूचकांक वर्ग उदाहरण और क्वेरी विधि के लिए सूचकांक और क्वेरी ऑब्जेक्ट पास करें।

बेहतर समझने के लिए निम्नलिखित कोड देखें -

DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient ( 
   new ProfileCredentialsProvider()));
   
Table table = dynamoDB.getTable("ClimateInfo"); 
Index index = table.getIndex("WindIndex");  
QuerySpec spec = new QuerySpec() 
   .withKeyConditionExpression("#d = :v_date and Wind = :v_wind") 
   .withNameMap(new NameMap() 
   .with("#d", "Date"))
   .withValueMap(new ValueMap() 
   .withString(":v_date","2016-05-15") 
   .withNumber(":v_wind",0));
   
ItemCollection<QueryOutcome> items = index.query(spec);
Iterator<Item> iter = items.iterator();

while (iter.hasNext()) {
   System.out.println(iter.next().toJSONPretty()); 
}

निम्नलिखित कार्यक्रम बेहतर समझ के लिए एक बड़ा उदाहरण है -

Note- निम्न प्रोग्राम पहले से बनाए गए डेटा स्रोत को मान सकता है। निष्पादित करने का प्रयास करने से पहले, सहायक पुस्तकालयों का अधिग्रहण करें और आवश्यक डेटा स्रोत बनाएं (आवश्यक विशेषताओं के साथ टेबल, या अन्य संदर्भित स्रोत)।

यह उदाहरण ग्रहण आईडीई, एक एडब्ल्यूएस क्रेडेंशियल फ़ाइल और एडब्ल्यूएस टूलकिट को एक्लिप्स एईए जावा प्रोजेक्ट के भीतर भी उपयोग करता है।

import java.util.ArrayList;
import java.util.Iterator;

import com.amazonaws.auth.profile.ProfileCredentialsProvider;
import com.amazonaws.services.dynamodbv2.AmazonDynamoDBClient;
import com.amazonaws.services.dynamodbv2.document.DynamoDB;
import com.amazonaws.services.dynamodbv2.document.Index;
import com.amazonaws.services.dynamodbv2.document.Item;
import com.amazonaws.services.dynamodbv2.document.ItemCollection;
import com.amazonaws.services.dynamodbv2.document.QueryOutcome;
import com.amazonaws.services.dynamodbv2.document.Table;
import com.amazonaws.services.dynamodbv2.document.spec.QuerySpec;
import com.amazonaws.services.dynamodbv2.document.utils.ValueMap;

import com.amazonaws.services.dynamodbv2.model.AttributeDefinition;
import com.amazonaws.services.dynamodbv2.model.CreateTableRequest;
import com.amazonaws.services.dynamodbv2.model.GlobalSecondaryIndex;
import com.amazonaws.services.dynamodbv2.model.KeySchemaElement;
import com.amazonaws.services.dynamodbv2.model.KeyType;
import com.amazonaws.services.dynamodbv2.model.Projection;
import com.amazonaws.services.dynamodbv2.model.ProvisionedThroughput;

public class GlobalSecondaryIndexSample {  
   static DynamoDB dynamoDB = new DynamoDB(new AmazonDynamoDBClient ( 
      new ProfileCredentialsProvider()));  
   public static String tableName = "Bugs";   
   public static void main(String[] args) throws Exception {  
      createTable(); 
      queryIndex("CreationDateIndex"); 
      queryIndex("NameIndex"); 
      queryIndex("DueDateIndex"); 
   }
   public static void createTable() {  
      // Attributes 
      ArrayList<AttributeDefinition> attributeDefinitions = new 
         ArrayList<AttributeDefinition>();  
      attributeDefinitions.add(new AttributeDefinition()
         .withAttributeName("BugID") 
         .withAttributeType("S")); 
         
      attributeDefinitions.add(new AttributeDefinition() 
         .withAttributeName("Name")
         .withAttributeType("S"));
         
      attributeDefinitions.add(new AttributeDefinition() 
         .withAttributeName("CreationDate")
         .withAttributeType("S"));
         
      attributeDefinitions.add(new AttributeDefinition() 
         .withAttributeName("DueDate") 
         .withAttributeType("S"));
         
      // Table Key schema
      ArrayList<KeySchemaElement> tableKeySchema = new ArrayList<KeySchemaElement>(); 
      tableKeySchema.add (new KeySchemaElement() 
         .withAttributeName("BugID") 
         .withKeyType(KeyType.HASH));              //Partition key 
      
      tableKeySchema.add (new KeySchemaElement() 
         .withAttributeName("Name") 
         .withKeyType(KeyType.RANGE));             //Sort key
         
      // Indexes' initial provisioned throughput
      ProvisionedThroughput ptIndex = new ProvisionedThroughput()
         .withReadCapacityUnits(1L)
         .withWriteCapacityUnits(1L);
         
      // CreationDateIndex 
      GlobalSecondaryIndex creationDateIndex = new GlobalSecondaryIndex() 
         .withIndexName("CreationDateIndex") 
         .withProvisionedThroughput(ptIndex) 
         .withKeySchema(new KeySchemaElement() 
         .withAttributeName("CreationDate") 
         .withKeyType(KeyType.HASH),               //Partition key 
         new KeySchemaElement()
         .withAttributeName("BugID") 
         .withKeyType(KeyType.RANGE))              //Sort key 
         .withProjection(new Projection() 
         .withProjectionType("INCLUDE") 
         .withNonKeyAttributes("Description", "Status"));
         
      // NameIndex 
      GlobalSecondaryIndex nameIndex = new GlobalSecondaryIndex() 
         .withIndexName("NameIndex") 
         .withProvisionedThroughput(ptIndex) 
         .withKeySchema(new KeySchemaElement()  
         .withAttributeName("Name")  
         .withKeyType(KeyType.HASH),                  //Partition key 
         new KeySchemaElement()  
         .withAttributeName("BugID")  
         .withKeyType(KeyType.RANGE))                 //Sort key 
         .withProjection(new Projection() 
         .withProjectionType("KEYS_ONLY"));
         
      // DueDateIndex 
      GlobalSecondaryIndex dueDateIndex = new GlobalSecondaryIndex() 
         .withIndexName("DueDateIndex") 
         .withProvisionedThroughput(ptIndex) 
         .withKeySchema(new KeySchemaElement() 
         .withAttributeName("DueDate") 
         .withKeyType(KeyType.HASH))               //Partition key 
         .withProjection(new Projection() 
         .withProjectionType("ALL"));
         
      CreateTableRequest createTableRequest = new CreateTableRequest() 
         .withTableName(tableName) 
         .withProvisionedThroughput( new ProvisionedThroughput() 
         .withReadCapacityUnits( (long) 1) 
         .withWriteCapacityUnits( (long) 1)) 
         .withAttributeDefinitions(attributeDefinitions)
         .withKeySchema(tableKeySchema)
         .withGlobalSecondaryIndexes(creationDateIndex, nameIndex, dueDateIndex);  
         System.out.println("Creating " + tableName + "..."); 
         dynamoDB.createTable(createTableRequest);  
      
      // Pause for active table state 
      System.out.println("Waiting for ACTIVE state of " + tableName); 
      try { 
         Table table = dynamoDB.getTable(tableName); 
         table.waitForActive(); 
      } catch (InterruptedException e) { 
         e.printStackTrace(); 
      } 
   }
   public static void queryIndex(String indexName) { 
      Table table = dynamoDB.getTable(tableName);  
      System.out.println 
      ("\n*****************************************************\n"); 
      System.out.print("Querying index " + indexName + "...");  
      Index index = table.getIndex(indexName);  
      ItemCollection<QueryOutcome> items = null; 
      QuerySpec querySpec = new QuerySpec();  
      
      if (indexName == "CreationDateIndex") { 
         System.out.println("Issues filed on 2016-05-22"); 
         querySpec.withKeyConditionExpression("CreationDate = :v_date and begins_with
            (BugID, :v_bug)") 
            .withValueMap(new ValueMap() 
            .withString(":v_date","2016-05-22")
            .withString(":v_bug","A-")); 
         items = index.query(querySpec); 
      } else if (indexName == "NameIndex") { 
         System.out.println("Compile error"); 
         querySpec.withKeyConditionExpression("Name = :v_name and begins_with
            (BugID, :v_bug)") 
            .withValueMap(new ValueMap() 
            .withString(":v_name","Compile error") 
            .withString(":v_bug","A-")); 
         items = index.query(querySpec); 
      } else if (indexName == "DueDateIndex") { 
         System.out.println("Items due on 2016-10-15"); 
         querySpec.withKeyConditionExpression("DueDate = :v_date") 
         .withValueMap(new ValueMap() 
         .withString(":v_date","2016-10-15")); 
         items = index.query(querySpec); 
      } else { 
         System.out.println("\nInvalid index name"); 
         return; 
      }  
      Iterator<Item> iterator = items.iterator(); 
      System.out.println("Query: getting result..."); 
      
      while (iterator.hasNext()) { 
         System.out.println(iterator.next().toJSONPretty()); 
      } 
   } 
}