SlideShare a Scribd company logo
www.dotnetconf.net
.NET Conf 2019 - Indexing and searching NuGet.org with Azure Functions and Search
“Find this type
on NuGet.org”
“Find this type on NuGet.org”
In ReSharper and Rider
Search for namespaces
& types that are not yet referenced
“Find this type on NuGet.org”
Introduced in ReSharper 9
(2015 - https://www.jetbrains.com/resharper/whatsnew/whatsnew_9.html)
Consists of
ReSharper functionality
A service that indexes packages and powers search
Azure Cloud Service (Web and Worker role)
Indexer uses NuGet OData feed
https://www.nuget.org/api/v2/Packages?$select=Id,Version,NormalizedVersion,LastEdited,Published&$
orderby=LastEdited%20desc&$filter=LastEdited%20gt%20datetime%272012-01-01%27
NuGet over time...
https://twitter.com/controlflow/status/1067724815958777856 https://github.com/NuGet/Announcements/issues/37
NuGet server-side API
V3 Protocol
JSON based
A “resource provider” of various endpoints per purpose
Search
Autocomplete
Package restore
Catalog (NuGet.org only) – append-only event log
…
https://api.nuget.org/v3/index.json (code in https://github.com/NuGet/NuGet.Services.Metadata)
Catalog seems interesting!
Append-only stream of mutations on NuGet.org
Updates (add/update) and Deletes
Chronological
Can continue where left off (uses a timestamp cursor)
Can restore NuGet.org to a given point in time
Structure
Root https://api.nuget.org/v3/catalog0/index.json
+ Page https://api.nuget.org/v3/catalog0/page0.json
+ Leaf https://api.nuget.org/v3/catalog0/data/2015.02.01.06.22.45/adam.jsgenerator.1.1.0.json
NuGet.org catalog
demo
“Find this type on NuGet.org”
Refactor from using OData to using V3?
Mostly done, one thing missing: download counts (using search now)
https://github.com/NuGet/NuGetGallery/issues/3532
Build a new version?
Welcome to this talk 
Building a new version
What do we need?
Watch the NuGet.org catalog for package changes
For every package change
Scan all assemblies
Store relation between package id+version and namespace+type
API compatible with all ReSharper and Rider versions
What do we need?
Watch the NuGet.org catalog for package changes periodic check
For every package change based on a queue
Scan all assemblies
Store relation between package id+version and namespace+type
API compatible with all ReSharper and Rider versions always up, flexible scale
Sounds like Azure functions!
NuGet.org catalog Watch catalog
Index command
Find type API
Find namespace API
Search index
Index package
Raw .nupkg
Index as JSON
Download packageDownload command
Collecting from catalog
demo
Functions best practices
@PaulDJohnston https://medium.com/@PaulDJohnston/serverless-best-practices-b3c97d551535
Each function should do only one thing
Easier error handling & scaling
Learn to use messages and queues
Asynchronous means of communicating, helps scale and avoid direct coupling
...
Bindings
Help a function do only one thing
Trigger, provide input/output
Function code bridges those
Build your own!*
SQL Server binding
Dropbox binding
...
NuGet Catalog
*Custom triggers not officially supported (yet?)
Trigger Input Output
Timer ✔
HTTP ✔ ✔
Blob ✔ ✔ ✔
Queue ✔ ✔
Table ✔ ✔
Service Bus ✔ ✔
EventHub ✔ ✔
EventGrid ✔
CosmosDB ✔ ✔ ✔
IoT Hub ✔
SendGrid, Twilio ✔
... ✔
Creating a trigger
binding
demo
Indexing
Download .nupkg and open (it’s a ZIP file)
Reflecting on assemblies
System.Reflection.Metadata
Access Portable Executable (PE) metadata in assembly without loading into process
Store relation between package id+version and namespace+type
Azure Search
Indexing packages
demo
Next steps…
✓ Watch the NuGet.org catalog for package changes
✓ For every package change
Scan all assemblies
Store relation between package id+version and namespace+type
API compatible with all ReSharper and Rider versions
Blog post with full story and implementation
https://bit.ly/2lzba32
👀
Making search work
with ReSharper and Rider
demo
Learnings
NuGet.org catalog Watch catalog
Index command
Find type API
Find namespace API
Search index
Index package
Raw .nupkg
Index as JSON
Download packageDownload command
Learnings
Functions
Collect changes from NuGet catalog
Download binaries
Index binaries using PE Header
Make search index available in API
Trigger, input and output bindings
Each function should do only one thing
NuGet.org catalog Watch catalog
Index command
Find type API
Find namespace API
Search index
Index package
Raw .nupkg
Index as JSON
Download packageDownload command
Learnings
All our functions can scale (and fail)
independently
Full index in May 2019 took ~12h on 2 B1 instances
Can be faster on more CPU’s
~ 2.1mio unique packages in the catalog
~ 8 400 catalog pages
with ~ 4 200 000 catalog leaves
NuGet.org catalog Watch catalog
Index command
Find type API
Find namespace API
Search index
Index package
Raw .nupkg
Index as JSON
Download packageDownload command
Learnings
Deploy in separate function apps for cost, performance & fault boundaries
Trigger binding collects all the time – small reserved instance
Other functions on-demand – consumption plan
Bindings
Very nice programming model
https://bit.ly/2lzba32

More Related Content

PPTX
Indexing and searching NuGet.org with Azure Functions and Search - .NET fwday...
PDF
Maarten Balliauw "Indexing and searching NuGet.org with Azure Functions and S...
PDF
Mastering Grails 3 Plugins - Greach 2016
PDF
PPTX
MySQL Slow Query log Monitoring using Beats & ELK
PDF
Streaming using Kafka Flink & Elasticsearch
PDF
Spark Day 2017- Spark 의 과거, 현재, 미래
PDF
Introduction to Octopress at DRUG
Indexing and searching NuGet.org with Azure Functions and Search - .NET fwday...
Maarten Balliauw "Indexing and searching NuGet.org with Azure Functions and S...
Mastering Grails 3 Plugins - Greach 2016
MySQL Slow Query log Monitoring using Beats & ELK
Streaming using Kafka Flink & Elasticsearch
Spark Day 2017- Spark 의 과거, 현재, 미래
Introduction to Octopress at DRUG

What's hot (20)

PDF
The End of Dinosaurs happened because of [a] Meteor
PDF
pio_present
PDF
Multiple django applications on a single server with nginx
PPTX
Dockercompose
PDF
Jenkins and Groovy
PDF
Deploy Python apps in 5 min with a PaaS
PDF
Real-time search in Drupal. Meet Elasticsearch
PPTX
Introduction to MongoDB
PDF
Apache spark undocumented extensions
PDF
Kubernetes extensibility: crd & operators
PDF
Android Libs - Retrofit
PDF
Cassandra Summit 2014: Highly Scalable Web Application in the Cloud with Cass...
PDF
Helpful pre commit hooks for Python and Django
PDF
Карманный PaaS с Dokku (Александр Белецкий)
PPTX
Retrofit Technology Overview by Cumulations Technologies
PDF
Grails 1.4.0.M1 メモLT
PDF
Nicola Iarocci - Git stories from the front line - Codemotion Milan 2017
PDF
OpenStack API's and WSGI
PDF
GitOps: Stop, collaborate and deploy | DevNation Tech Talk
PDF
Puppet Camp Paris 2015: Continuous Integration of Puppet Code (Intermediate)
The End of Dinosaurs happened because of [a] Meteor
pio_present
Multiple django applications on a single server with nginx
Dockercompose
Jenkins and Groovy
Deploy Python apps in 5 min with a PaaS
Real-time search in Drupal. Meet Elasticsearch
Introduction to MongoDB
Apache spark undocumented extensions
Kubernetes extensibility: crd & operators
Android Libs - Retrofit
Cassandra Summit 2014: Highly Scalable Web Application in the Cloud with Cass...
Helpful pre commit hooks for Python and Django
Карманный PaaS с Dokku (Александр Белецкий)
Retrofit Technology Overview by Cumulations Technologies
Grails 1.4.0.M1 メモLT
Nicola Iarocci - Git stories from the front line - Codemotion Milan 2017
OpenStack API's and WSGI
GitOps: Stop, collaborate and deploy | DevNation Tech Talk
Puppet Camp Paris 2015: Continuous Integration of Puppet Code (Intermediate)
Ad

Similar to .NET Conf 2019 - Indexing and searching NuGet.org with Azure Functions and Search (20)

PPTX
Indexing and searching NuGet.org with Azure Functions and Search - Cloud Deve...
PPTX
NDC Oslo 2019 - Indexing and searching NuGet.org with Azure Functions and Search
PPTX
CloudBurst 2019 - Indexing and searching NuGet.org with Azure Functions and S...
PPTX
NuGet 3.0 - Transitioning from OData to JSON-LD
PPTX
Understanding NuGet implementation for Enterprises
PPTX
NuGet beyond Hello World - DotNext Piter 2017
PPTX
ConFoo - NuGet beyond Hello World
PPTX
Using nu get the way you should svcc
PPTX
Using NuGet the way you should - TechDays NL 2014
PPTX
Intro to NuGet
PDF
Nuget is easier than you think and you should be using it as both a consumer ...
PPTX
NuGet (anti-)patterns - Tales from the Trenches
PPTX
Tamir Dresher - DotNet 7 What's new.pptx
PPTX
Evolution of NuGet
PPTX
NuGet Packages Presentation (DoT NeT).pptx
PPTX
Azure serverless computing
PPTX
Windows 8 für .net Entwickler
PPTX
An overview of the NuGet ecosystem - Mobel.io
PPTX
Controlling Component Chaos with NuGet and Versioning
PPTX
NuGet (Anti-)Patterns - Tales from the Trenches
Indexing and searching NuGet.org with Azure Functions and Search - Cloud Deve...
NDC Oslo 2019 - Indexing and searching NuGet.org with Azure Functions and Search
CloudBurst 2019 - Indexing and searching NuGet.org with Azure Functions and S...
NuGet 3.0 - Transitioning from OData to JSON-LD
Understanding NuGet implementation for Enterprises
NuGet beyond Hello World - DotNext Piter 2017
ConFoo - NuGet beyond Hello World
Using nu get the way you should svcc
Using NuGet the way you should - TechDays NL 2014
Intro to NuGet
Nuget is easier than you think and you should be using it as both a consumer ...
NuGet (anti-)patterns - Tales from the Trenches
Tamir Dresher - DotNet 7 What's new.pptx
Evolution of NuGet
NuGet Packages Presentation (DoT NeT).pptx
Azure serverless computing
Windows 8 für .net Entwickler
An overview of the NuGet ecosystem - Mobel.io
Controlling Component Chaos with NuGet and Versioning
NuGet (Anti-)Patterns - Tales from the Trenches
Ad

More from Maarten Balliauw (20)

PPTX
Bringing nullability into existing code - dammit is not the answer.pptx
PPTX
Nerd sniping myself into a rabbit hole... Streaming online audio to a Sonos s...
PPTX
Building a friendly .NET SDK to connect to Space
PPTX
Microservices for building an IDE - The innards of JetBrains Rider - NDC Oslo...
PPTX
NDC Sydney 2019 - Microservices for building an IDE – The innards of JetBrain...
PPTX
JetBrains Australia 2019 - Exploring .NET’s memory management – a trip down m...
PPTX
Approaches for application request throttling - Cloud Developer Days Poland
PPTX
Approaches for application request throttling - dotNetCologne
PPTX
CodeStock - Exploring .NET memory management - a trip down memory lane
PPTX
ConFoo Montreal - Microservices for building an IDE - The innards of JetBrain...
PPTX
ConFoo Montreal - Approaches for application request throttling
PPTX
Microservices for building an IDE – The innards of JetBrains Rider - TechDays...
PPTX
JetBrains Day Seoul - Exploring .NET’s memory management – a trip down memory...
PPTX
DotNetFest - Let’s refresh our memory! Memory management in .NET
PPTX
VISUG - Approaches for application request throttling
PPTX
What is going on - Application diagnostics on Azure - TechDays Finland
PPTX
ConFoo - Exploring .NET’s memory management – a trip down memory lane
PPTX
Approaches to application request throttling
PPTX
Exploring .NET memory management (iSense)
PPTX
Exploring .NET memory management - JetBrains webinar
Bringing nullability into existing code - dammit is not the answer.pptx
Nerd sniping myself into a rabbit hole... Streaming online audio to a Sonos s...
Building a friendly .NET SDK to connect to Space
Microservices for building an IDE - The innards of JetBrains Rider - NDC Oslo...
NDC Sydney 2019 - Microservices for building an IDE – The innards of JetBrain...
JetBrains Australia 2019 - Exploring .NET’s memory management – a trip down m...
Approaches for application request throttling - Cloud Developer Days Poland
Approaches for application request throttling - dotNetCologne
CodeStock - Exploring .NET memory management - a trip down memory lane
ConFoo Montreal - Microservices for building an IDE - The innards of JetBrain...
ConFoo Montreal - Approaches for application request throttling
Microservices for building an IDE – The innards of JetBrains Rider - TechDays...
JetBrains Day Seoul - Exploring .NET’s memory management – a trip down memory...
DotNetFest - Let’s refresh our memory! Memory management in .NET
VISUG - Approaches for application request throttling
What is going on - Application diagnostics on Azure - TechDays Finland
ConFoo - Exploring .NET’s memory management – a trip down memory lane
Approaches to application request throttling
Exploring .NET memory management (iSense)
Exploring .NET memory management - JetBrains webinar

Recently uploaded (20)

PDF
Encapsulation theory and applications.pdf
PPTX
A Presentation on Artificial Intelligence
PDF
Network Security Unit 5.pdf for BCA BBA.
PPTX
20250228 LYD VKU AI Blended-Learning.pptx
PDF
Building Integrated photovoltaic BIPV_UPV.pdf
PDF
Modernizing your data center with Dell and AMD
PDF
Encapsulation_ Review paper, used for researhc scholars
PDF
NewMind AI Monthly Chronicles - July 2025
PDF
Review of recent advances in non-invasive hemoglobin estimation
PPTX
KOM of Painting work and Equipment Insulation REV00 update 25-dec.pptx
PPTX
Digital-Transformation-Roadmap-for-Companies.pptx
PPTX
Detection-First SIEM: Rule Types, Dashboards, and Threat-Informed Strategy
PDF
Reach Out and Touch Someone: Haptics and Empathic Computing
PPTX
MYSQL Presentation for SQL database connectivity
PPTX
Effective Security Operations Center (SOC) A Modern, Strategic, and Threat-In...
PDF
Unlocking AI with Model Context Protocol (MCP)
PDF
NewMind AI Weekly Chronicles - August'25 Week I
PDF
Empathic Computing: Creating Shared Understanding
PDF
7 ChatGPT Prompts to Help You Define Your Ideal Customer Profile.pdf
PDF
Peak of Data & AI Encore- AI for Metadata and Smarter Workflows
Encapsulation theory and applications.pdf
A Presentation on Artificial Intelligence
Network Security Unit 5.pdf for BCA BBA.
20250228 LYD VKU AI Blended-Learning.pptx
Building Integrated photovoltaic BIPV_UPV.pdf
Modernizing your data center with Dell and AMD
Encapsulation_ Review paper, used for researhc scholars
NewMind AI Monthly Chronicles - July 2025
Review of recent advances in non-invasive hemoglobin estimation
KOM of Painting work and Equipment Insulation REV00 update 25-dec.pptx
Digital-Transformation-Roadmap-for-Companies.pptx
Detection-First SIEM: Rule Types, Dashboards, and Threat-Informed Strategy
Reach Out and Touch Someone: Haptics and Empathic Computing
MYSQL Presentation for SQL database connectivity
Effective Security Operations Center (SOC) A Modern, Strategic, and Threat-In...
Unlocking AI with Model Context Protocol (MCP)
NewMind AI Weekly Chronicles - August'25 Week I
Empathic Computing: Creating Shared Understanding
7 ChatGPT Prompts to Help You Define Your Ideal Customer Profile.pdf
Peak of Data & AI Encore- AI for Metadata and Smarter Workflows

.NET Conf 2019 - Indexing and searching NuGet.org with Azure Functions and Search

  • 3. “Find this type on NuGet.org”
  • 4. “Find this type on NuGet.org” In ReSharper and Rider Search for namespaces & types that are not yet referenced
  • 5. “Find this type on NuGet.org” Introduced in ReSharper 9 (2015 - https://www.jetbrains.com/resharper/whatsnew/whatsnew_9.html) Consists of ReSharper functionality A service that indexes packages and powers search Azure Cloud Service (Web and Worker role) Indexer uses NuGet OData feed https://www.nuget.org/api/v2/Packages?$select=Id,Version,NormalizedVersion,LastEdited,Published&$ orderby=LastEdited%20desc&$filter=LastEdited%20gt%20datetime%272012-01-01%27
  • 6. NuGet over time... https://twitter.com/controlflow/status/1067724815958777856 https://github.com/NuGet/Announcements/issues/37
  • 8. V3 Protocol JSON based A “resource provider” of various endpoints per purpose Search Autocomplete Package restore Catalog (NuGet.org only) – append-only event log … https://api.nuget.org/v3/index.json (code in https://github.com/NuGet/NuGet.Services.Metadata)
  • 9. Catalog seems interesting! Append-only stream of mutations on NuGet.org Updates (add/update) and Deletes Chronological Can continue where left off (uses a timestamp cursor) Can restore NuGet.org to a given point in time Structure Root https://api.nuget.org/v3/catalog0/index.json + Page https://api.nuget.org/v3/catalog0/page0.json + Leaf https://api.nuget.org/v3/catalog0/data/2015.02.01.06.22.45/adam.jsgenerator.1.1.0.json
  • 11. “Find this type on NuGet.org” Refactor from using OData to using V3? Mostly done, one thing missing: download counts (using search now) https://github.com/NuGet/NuGetGallery/issues/3532 Build a new version? Welcome to this talk 
  • 12. Building a new version
  • 13. What do we need? Watch the NuGet.org catalog for package changes For every package change Scan all assemblies Store relation between package id+version and namespace+type API compatible with all ReSharper and Rider versions
  • 14. What do we need? Watch the NuGet.org catalog for package changes periodic check For every package change based on a queue Scan all assemblies Store relation between package id+version and namespace+type API compatible with all ReSharper and Rider versions always up, flexible scale
  • 15. Sounds like Azure functions! NuGet.org catalog Watch catalog Index command Find type API Find namespace API Search index Index package Raw .nupkg Index as JSON Download packageDownload command
  • 17. Functions best practices @PaulDJohnston https://medium.com/@PaulDJohnston/serverless-best-practices-b3c97d551535 Each function should do only one thing Easier error handling & scaling Learn to use messages and queues Asynchronous means of communicating, helps scale and avoid direct coupling ...
  • 18. Bindings Help a function do only one thing Trigger, provide input/output Function code bridges those Build your own!* SQL Server binding Dropbox binding ... NuGet Catalog *Custom triggers not officially supported (yet?) Trigger Input Output Timer ✔ HTTP ✔ ✔ Blob ✔ ✔ ✔ Queue ✔ ✔ Table ✔ ✔ Service Bus ✔ ✔ EventHub ✔ ✔ EventGrid ✔ CosmosDB ✔ ✔ ✔ IoT Hub ✔ SendGrid, Twilio ✔ ... ✔
  • 20. Indexing Download .nupkg and open (it’s a ZIP file) Reflecting on assemblies System.Reflection.Metadata Access Portable Executable (PE) metadata in assembly without loading into process Store relation between package id+version and namespace+type Azure Search
  • 22. Next steps… ✓ Watch the NuGet.org catalog for package changes ✓ For every package change Scan all assemblies Store relation between package id+version and namespace+type API compatible with all ReSharper and Rider versions Blog post with full story and implementation https://bit.ly/2lzba32 👀
  • 23. Making search work with ReSharper and Rider demo
  • 24. Learnings NuGet.org catalog Watch catalog Index command Find type API Find namespace API Search index Index package Raw .nupkg Index as JSON Download packageDownload command
  • 25. Learnings Functions Collect changes from NuGet catalog Download binaries Index binaries using PE Header Make search index available in API Trigger, input and output bindings Each function should do only one thing NuGet.org catalog Watch catalog Index command Find type API Find namespace API Search index Index package Raw .nupkg Index as JSON Download packageDownload command
  • 26. Learnings All our functions can scale (and fail) independently Full index in May 2019 took ~12h on 2 B1 instances Can be faster on more CPU’s ~ 2.1mio unique packages in the catalog ~ 8 400 catalog pages with ~ 4 200 000 catalog leaves NuGet.org catalog Watch catalog Index command Find type API Find namespace API Search index Index package Raw .nupkg Index as JSON Download packageDownload command
  • 27. Learnings Deploy in separate function apps for cost, performance & fault boundaries Trigger binding collects all the time – small reserved instance Other functions on-demand – consumption plan Bindings Very nice programming model

Editor's Notes

  • #6: Show feature in action in Visual Studio (and show you can see basic metadata etc.)
  • #7: Copied in 2017 in VS - https://www.hanselman.com/blog/VisualStudio2017CanAutomaticallyRecommendNuGetPackagesForUnknownTypes.aspx Demo the feed quickly?
  • #8: More and more packages OData feed slow to query & scheduled for deprecation Is there a better way?
  • #10: Demo: click around in the API to show some base things
  • #12: Raw API - click around in the API to show some base things, explain how a cursor could go over it Root https://api.nuget.org/v3/catalog0/index.json Page https://api.nuget.org/v3/catalog0/page0.json Leaf https://api.nuget.org/v3/catalog0/data/2015.02.01.06.22.45/adam.jsgenerator.1.1.0.json Explain CatalogDump NuGet.Protocol.Catalog comes from GitHub CatalogProcessor feches all pages between min and max timestamp My implementation BatchCatalogProcessor fetches multiple pages at the same time and build a “latest state” – much faster! Fetches leaves, for every leaf calls into a simple method Much faster, easy to pause (keep track of min/max timestamp)
  • #17: Will use storage queues n demo’s to be able to run things locally. Ideally use SB topics or event grid (transactional)
  • #18: Create a new TimerTrigger function We will need a function to index things from NuGet Timer will trigger every X amount of time Timer provides last timestamp and next timestamp, so we can run our collector for that period Snippet: demo-timertrigger Mention HttpClient not used correctly: not disposed, so will starve TCP connections at some point Go over code example and run it var httpClient = new HttpClient(); var cursor = new InMemoryCursor(timer.ScheduleStatus?.Last ?? DateTimeOffset.UtcNow); var processor = new CatalogProcessor( cursor, new CatalogClient(httpClient, new NullLogger<CatalogClient>()), new DelegatingCatalogLeafProcessor( added => { log.LogInformation("[ADDED] " + added.PackageId + "@" + added.PackageVersion); return Task.FromResult(true); }, deleted => { log.LogInformation("[DELETED] " + deleted.PackageId + "@" + deleted.PackageVersion); return Task.FromResult(true); }), new CatalogProcessorSettings { MinCommitTimestamp = timer.ScheduleStatus?.Last ?? DateTimeOffset.UtcNow, MaxCommitTimestamp = timer.ScheduleStatus?.Next ?? DateTimeOffset.UtcNow, ServiceIndexUrl = "https://api.nuget.org/v3/index.json" }, new NullLogger<CatalogProcessor>()); await processor.ProcessAsync(CancellationToken.None);
  • #19: Each function should only do one thing! We are violating this.
  • #20: https://github.com/thinktecture/azure-functions-extensibility
  • #21: Go over Approach2 code Show this is MUCH simpler – trigger binding that provides input, queue output bindign to write that input to a queue Let’s go over what it takes to build a trigger binding NuGetCatalogTriggerAttribute – the data needed for the trigger to work – go over properties and attributes Hooking it up requires a binding configuration – NuGetCatalogTriggerExtensionConfigProvider It says: if you see this specific binding, register it as a trigger that maps to some provider So we need that provider – NuGetCatalogTriggerAttributeBindingProvider Provider is there to create an object that provides data. In our case we need to store the NuGet catalog timestamp cursor, so we do that on storage, and then return the actual binding – NuGetCatalogTriggerBinding In NuGetCatalogTriggerBinding, we have to specify how data can be bound. What if I use a differnt type of object than PackageOperation? What if someone used a node.js or Python function instead of .NET. Need to define the shape of the data our trigger provides. PackageOperationValueProvider is also interesting, this provides data shown in the portal diagnostics CreateListenerAsync is where the actual triger code will be created – NuGetCatalogListener NuGetCatalogListener uses the BatchCatalogProcessor we had previously, and when a package is added or deleted it will call into the injected ITriggeredFunctionExecutor ITriggeredFunctionExecutor is Azure Functions framework specific, but it’s the glue that will clal into our function with the data we provide Note StartAsync/StopAsync where you can add startup/shutdown code ONE THING LEFT THAT IS NOT DOCUMENTED – Startup.cs to register the binding. And since we are in a different class library, also need Microsoft.Azure.WebJobs.Extensions referenced to generate \bin\Debug\netcoreapp2.1\bin\extensions.json As a result our code is now MUCH cleaner, show it again and maybe also show it in action Mention [Singleton(Mode = SingletonMode.Listener)] – we need to ensure this binding only runs single-instance (cursor clashes otherwise). This is due to ho the catalog works, parallel processing is harder to do. But we can fix that by scaling the Indexer later on. Show Approach3 PopulateQueueAndTable Same code, but a bit more production worthy Sending data to two queues (indexing and downloading) Storing data in a table (and yes, violating “do one thing” again but I call it architectural freedom)
  • #23: Go over Approach3 code PackageIndexerWithCustomBinding is mostly the same code One difference: it uses the [AzureSearchIndex] binding to write add/delete operations to the index instead Go over how it works. Again, an attribute with settings – AzureSearchIndexAttribute Also a configuration that registers the binding as an output binding using BindToCollector – AzureSearchExtensionConfigProvider Now, what’s this OpenType? It’s some sort of dynamic type. If we want to create an AzureSearch output binding, we better support more than just our PackageDocument use case! So we need a collector builder that can create the actual binding implementation based on the real type requested by our function parameter – AzureSearchAsyncCollectorBuilder In AzureSearchAsyncCollectorBuilder, we do that. Very simple bootstrap code in this case, but could be more complex depending on the type of binding you are creating. Our AzureSearchAsyncCollector uses the attribute to check for Azure Search connection details, as well as the type of operation we expect it to handle. Why not all? Well, IAsyncCollector only has Add and Flush. Note: add called manually, flush at function complete – could use flush to send things in a batch... Code itself pretty straightforward. On Add, we add an action to search. With a retry in case the index does not exist – we then create it. Creation code kind of interesting as we use some reflection in case we specify a given type of coument to index. Why? Cause when we do Upserts, we may want to update just one or two properties, and can use a different Dto in that case (but still have the index shaped to the full document shape) Run when time left, but nothing fancy here...
  • #25: Go over Web code RunFindTypeApiAsync and RunFindNamespaceAsync Both use “name” as their query parameter to search for RunInternalAsync does the heavy lifting Grabs other parameters Runs search, and collects several pages of results Why is this ForEachAsync there? Search index has multiple versions for every package id, yet ReSharper expects only the latest matching all parameters Azure Search has no group by / distinct by, so need to do this in memory. Doing it here by fetching a maximum number of results and doing the grouping manually. Use the collected data to build result. Add matching type names etc. Example requests: http://localhost:7071/api/v1/find-type?name=JsonConvert http://localhost:7071/api/v1/find-type?name=CamoServer&allowPrerelease=true&latestVersion=false https://nugettypesearch.azurewebsites.net/api/v1/find-type?name=JsonConvert In ReSharper (devenv /ReSharper.Internal, go to NuGet tool window, set base URL to https://nugettypesearch.azurewebsites.net/api/v1/) Write some code that uses JsonConvert / JObject and try it out.