public IEnumerable<Dashboard> Dashboards() { if (_dashboards != null && _dashboardLoaded.Add(_cacheLife) < DateTime.UtcNow) { return _dashboards; } var connection = new DocumentDbConnectionString(_connectionString); using (var client = new DocumentClient(connection.AccountEndpoint, connection.AccountKey)) { var database = client.ReadDatabaseFeedAsync().Result; var dashboardCollection = client.ReadDocumentCollectionFeedAsync( database.Single(x => x.Id == connection.DatabaseName).CollectionsLink).Result; var dashboards = client.ReadDocumentFeedAsync(dashboardCollection.Single(x => x.Id == "dashboards").DocumentsLink) .Result; _dashboards = dashboards.Select(x => new Dashboard { Title = x.GetPropertyValue<string>("Title"), Slug = x.GetPropertyValue<string>("Slug"), Sections = x.GetPropertyValue<IEnumerable<DashboardSection>>("Sections") }).OrderBy(x => x.Title); _dashboardLoaded = DateTime.UtcNow; return _dashboards; } }
/// <summary> /// Gets or creates the collections for the hash resolver. /// </summary> /// <param name="client">The DocumentDB client instance.</param> /// <param name="database">The database to use.</param> /// <param name="collectionIdPrefix">The prefix to use while creating collections.</param> /// <param name="spec">The specification/template to use to create collections.</param> /// <returns>The list of collection self links.</returns> private static List<string> GetCollections( DocumentClient client, Database database, string collectionIdPrefix, DocumentCollectionSpec spec) { var collections = new Dictionary<int, string>(); foreach (DocumentCollection collection in client.ReadDocumentCollectionFeedAsync(database.SelfLink).Result) { if (collection.Id.StartsWith(collectionIdPrefix)) { int collectionNumber = int.Parse(collection.Id.Replace(collectionIdPrefix, string.Empty)); collections[collectionNumber] = collection.SelfLink; } } // Return selflinks in ID order return collections.OrderBy(kvp => kvp.Key).Select(kvp => kvp.Value).ToList(); }
private static async Task LogDocumentCountsPerCollection(DocumentClient client, Database database) { foreach (DocumentCollection collection in await client.ReadDocumentCollectionFeedAsync(database.SelfLink)) { int numDocuments = 0; foreach (int document in client.CreateDocumentQuery<int>(collection.SelfLink, "SELECT VALUE 1 FROM ROOT", new FeedOptions { MaxItemCount = -1 })) { numDocuments++; } Console.WriteLine("Collection {0}: {1} documents", collection.Id, numDocuments); } }
static int Main(string[] args) { Options options = new Options(); if (!CommandLine.Parser.Default.ParseArguments(args, options)) { Console.WriteLine("Invalid arguments"); return 1; } using (DocumentClient client = new DocumentClient( new Uri(options.Endpoint), options.AuthKey, new ConnectionPolicy { ConnectionMode = ConnectionMode.Direct, ConnectionProtocol = Protocol.Tcp })) { Database database = client.CreateDatabaseQuery().Where(d => d.Id == options.Database).AsEnumerable().FirstOrDefault(); if (database == null) { Console.WriteLine("Cannot find database " + options.Database); return 2; } List<DocumentCollection> collections = client.ReadDocumentCollectionFeedAsync(database.SelfLink).Result.ToList(); int minimumRequiredCollections = Math.Max(options.NewCollections, options.CurrentCollections); if (collections.Count < minimumRequiredCollections) { Console.WriteLine("At least {0} collections must be pre-created", minimumRequiredCollections); return 3; } Console.WriteLine("Current distribution of documents across collections:"); LogDocumentCountsPerCollection(client, database).Wait(); Console.WriteLine(); HashPartitionResolver currentPartitionResolver = new HashPartitionResolver(options.PartitionKeyName, collections.Take(options.CurrentCollections).Select(c => c.SelfLink)); HashPartitionResolver nextPartitionResolver = new HashPartitionResolver(options.PartitionKeyName, collections.Take(options.NewCollections).Select(c => c.SelfLink)); int numberOfMovedDocuments = 0; Parallel.ForEach(currentPartitionResolver.CollectionLinks, collectionLink => { ResourceFeedReader<Document> feedReader = client.CreateDocumentFeedReader(collectionLink, new FeedOptions { MaxItemCount = -1 }); while (feedReader.HasMoreResults) { foreach (Document document in DocumentClientHelper.ExecuteWithRetryAsync<FeedResponse<Document>>(() => feedReader.ExecuteNextAsync()).Result) { object partitionKey = nextPartitionResolver.GetPartitionKey(document); string newCollectionLink = nextPartitionResolver.ResolveForCreate(partitionKey); if (newCollectionLink != collectionLink) { int count = Interlocked.Increment(ref numberOfMovedDocuments); DocumentClientHelper.ExecuteWithRetryAsync(() => client.DeleteDocumentAsync(document.SelfLink)).Wait(); DocumentClientHelper.ExecuteWithRetryAsync(() => client.CreateDocumentAsync(newCollectionLink, document)).Wait(); if (count % 100 == 0) { Console.WriteLine("Moved {0} documents between partitions", numberOfMovedDocuments); } } } } }); Console.WriteLine(); Console.WriteLine("Moved {0} documents between partitions.", numberOfMovedDocuments); Console.WriteLine(); Console.WriteLine("Current distribution of documents across collections:"); LogDocumentCountsPerCollection(client, database).Wait(); Console.WriteLine(); } return 0; }