SQL Server 2022: Exploring the DATE_BUCKET Function

๐Ÿ•’SQL Server 2022 introduces several new and exciting features, and one of the standout additions is the DATE_BUCKET function. This function allows you to group dates into fixed intervals, making it easier to analyze time-based data. In this blog, we’ll dive into how DATE_BUCKET works, using the JBDB database for our demonstrations. We’ll also explore a business use case to showcase the function’s practical applications.๐Ÿ•’

Business Use Case: Analyzing Customer Orders ๐Ÿ“Š

Imagine a retail company, “Retail Insights,” that wants to analyze customer order data to understand purchasing patterns over time. Specifically, the company wants to group orders into weekly intervals to identify trends and peak periods. Using the DATE_BUCKET function, we can efficiently bucketize order dates into weekly intervals and perform various analyses.

Setting Up the JBDB Database

First, let’s set up our sample database and table. We’ll create a database named JBDB and a table Orders to store our order data.

-- Create JBDB Database
CREATE DATABASE JBDB;
GO

-- Use JBDB Database
USE JBDB;
GO

-- Create Orders Table
CREATE TABLE Orders (
    OrderID INT PRIMARY KEY IDENTITY(1,1),
    CustomerID INT,
    OrderDate DATETIME,
    TotalAmount DECIMAL(10, 2)
);
GO

Inserting Sample Data ๐Ÿ“ฆ

Next, we’ll insert some sample data into the Orders table to simulate a few months of order history.

-- Insert Sample Data into Orders Table
INSERT INTO Orders (CustomerID, OrderDate, TotalAmount)
VALUES
(1, '2022-01-05', 250.00),
(2, '2022-01-12', 300.50),
(1, '2022-01-19', 450.00),
(3, '2022-01-25', 500.75),
(4, '2022-02-01', 320.00),
(5, '2022-02-08', 275.00),
(2, '2022-02-15', 150.25),
(3, '2022-02-22', 600.00),
(4, '2022-03-01', 350.00),
(5, '2022-03-08', 425.75);
GO

Using the DATE_BUCKET Function ๐Ÿ—“๏ธ

The DATE_BUCKET function simplifies the process of grouping dates into fixed intervals. Let’s see how it works by bucketing our orders into weekly intervals.

-- Group Orders into Weekly Intervals Using DATE_BUCKET
SELECT 
    CustomerID,
    OrderDate,
    TotalAmount,
    DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime)) AS OrderWeek
FROM Orders
ORDER BY OrderWeek;
GO

In the above query:

  • WEEK specifies the interval size.
  • 1 is the number of weeks per bucket.
  • OrderDate is the column containing the dates to be bucketed.
  • CAST('2022-01-01' AS datetime) is the reference date from which the intervals are calculated, cast to the datetime type to match OrderDate.

Analyzing Sales Trends ๐Ÿ“ˆ

Now that we have our orders grouped into weekly intervals, we can analyze sales trends, such as total sales per week.

-- Calculate Total Sales Per Week
SELECT 
    DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime)) AS OrderWeek,
    SUM(TotalAmount) AS TotalSales
FROM Orders
GROUP BY DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime))
ORDER BY OrderWeek;
GO

This query helps “Retail Insights” identify peak sales periods and trends over time. For example, they might find that certain weeks have consistently higher sales, prompting them to investigate further.

Grouping by Month

SELECT 
    CustomerID,
    OrderDate,
    TotalAmount,
    DATE_BUCKET(MONTH, 1, OrderDate, CAST('2022-01-01' AS datetime)) AS OrderMonth
FROM Orders
ORDER BY OrderMonth;
GO

Analyzing Orders Per Customer

SELECT 
    CustomerID,
    COUNT(OrderID) AS NumberOfOrders,
    SUM(TotalAmount) AS TotalSpent,
    DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime)) AS OrderWeek
FROM Orders
GROUP BY CustomerID, DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime))
ORDER BY OrderWeek;
GO

Counting Orders in Each Weekly Interval

This query counts the number of orders placed in each weekly interval.

-- Count Orders in Each Weekly Interval Using DATE_BUCKET
SELECT 
    DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime)) AS OrderWeek,
    COUNT(OrderID) AS NumberOfOrders
FROM Orders
GROUP BY DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime))
ORDER BY OrderWeek;
GO

Average Order Value per Week

Calculate the average value of orders in each weekly interval.

-- Calculate Average Order Value Per Week
SELECT 
    DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime)) AS OrderWeek,
    AVG(TotalAmount) AS AverageOrderValue
FROM Orders
GROUP BY DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime))
ORDER BY OrderWeek;
GO

Monthly Sales Analysis

Analyze total sales on a monthly basis.

-- Analyze Monthly Sales Using DATE_BUCKET
SELECT 
    DATE_BUCKET(MONTH, 1, OrderDate, CAST('2022-01-01' AS datetime)) AS OrderMonth,
    SUM(TotalAmount) AS MonthlySales
FROM Orders
GROUP BY DATE_BUCKET(MONTH, 1, OrderDate, CAST('2022-01-01' AS datetime))
ORDER BY OrderMonth;
GO

Identifying Peak Ordering Days

Identify the days with the highest total sales using daily buckets.

-- Identify Peak Ordering Days
SELECT 
    DATE_BUCKET(DAY, 1, OrderDate, CAST('2022-01-01' AS datetime)) AS OrderDay,
    SUM(TotalAmount) AS TotalSales
FROM Orders
GROUP BY DATE_BUCKET(DAY, 1, OrderDate, CAST('2022-01-01' AS datetime))
ORDER BY TotalSales DESC;
GO

Customer Order Frequency Analysis

Determine the frequency of orders for each customer on a weekly basis.

-- Customer Order Frequency Analysis Using DATE_BUCKET
SELECT 
    CustomerID,
    DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime)) AS OrderWeek,
    COUNT(OrderID) AS OrdersPerWeek
FROM Orders
GROUP BY CustomerID, DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime))
ORDER BY CustomerID, OrderWeek;
GO

Weekly Revenue Growth Rate

Calculate the weekly growth rate in sales revenue.

-- Calculate Weekly Revenue Growth Rate
WITH WeeklySales AS (
    SELECT 
        DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime)) AS OrderWeek,
        SUM(TotalAmount) AS WeeklySales
    FROM Orders
    GROUP BY DATE_BUCKET(WEEK, 1, OrderDate, CAST('2022-01-01' AS datetime))
)
SELECT 
    OrderWeek,
    WeeklySales,
    LAG(WeeklySales) OVER (ORDER BY OrderWeek) AS PreviousWeekSales,
    (WeeklySales - LAG(WeeklySales) OVER (ORDER BY OrderWeek)) / LAG(WeeklySales) OVER (ORDER BY OrderWeek) * 100 AS GrowthRate
FROM WeeklySales
ORDER BY OrderWeek;
GO

Orders Distribution Across Quarters

Analyze the distribution of orders across different quarters.

-- Distribution of Orders Across Quarters
SELECT 
    DATE_BUCKET(QUARTER, 1, OrderDate, CAST('2022-01-01' AS datetime)) AS OrderQuarter,
    COUNT(OrderID) AS NumberOfOrders
FROM Orders
GROUP BY DATE_BUCKET(QUARTER, 1, OrderDate, CAST('2022-01-01' AS datetime))
ORDER BY OrderQuarter;
GO

Business Insights ๐Ÿ’ก

Using the DATE_BUCKET function, “Retail Insights” can gain valuable insights into customer purchasing patterns:

  1. Identify Peak Periods: By analyzing weekly sales data, the company can pinpoint peak periods and prepare for increased demand.
  2. Marketing Strategies: Understanding customer behavior patterns helps in tailoring marketing strategies, such as promotions during slower periods.
  3. Inventory Management: Forecasting demand based on historical data enables better inventory planning and reduces stockouts or overstock situations.

Conclusion ๐ŸŽ‰

The DATE_BUCKET function in SQL Server 2022 is a powerful tool for time-based data analysis. It simplifies the process of grouping dates into intervals, making it easier to extract meaningful insights from your data. Whether you’re analyzing sales trends, customer behavior, or other time-sensitive information, DATE_BUCKET can help streamline your workflow and improve decision-making.

Feel free to try these examples in your own environment and explore the potential of DATE_BUCKET in your data analysis tasks! Happy querying! ๐Ÿš€

For more tutorials and tips on SQL Server, including performance tuning and database management, be sure to check out our JBSWiki YouTube channel.

Thank You,
Vivek Janakiraman

Disclaimer:
The views expressed on this blog are mine alone and do not reflect the views of my company or anyone else. All postings on this blog are provided โ€œAS ISโ€ with no warranties, and confers no rights.

Exploring SQL Server 2022’s Enhanced Support for Ordered Data in Window Functions

SQL Server 2022 has brought several exciting enhancements, especially for window functions. These improvements make it easier to work with ordered data, a common requirement in many business scenarios. In this blog, we will explore these new features using the JBDB database. We’ll start with a detailed business use case and demonstrate the improvements with practical T-SQL queries. Let’s dive in! ๐ŸŒŠ

Business Use Case: Sales Performance Analysis ๐Ÿ“Š

Imagine a company, JB Enterprises, which needs to analyze the sales performance of its sales representatives over time. The goal is to:

  1. Rank sales representatives based on their monthly sales.
  2. Calculate the running total of sales for each representative.
  3. Determine the difference in sales between the current month and the previous month.

To achieve this, we’ll use SQL Server 2022’s enhanced window functions.

Setting Up the JBDB Database ๐Ÿ› ๏ธ

First, let’s set up our JBDB database and create the necessary tables:

-- Create the JBDB database
CREATE DATABASE JBDB;
GO

-- Use the JBDB database
USE JBDB;
GO

-- Create the Sales table
CREATE TABLE Sales (
    SalesID INT PRIMARY KEY IDENTITY,
    SalesRepID INT,
    SalesRepName NVARCHAR(100),
    SaleDate DATE,
    SaleAmount DECIMAL(10, 2)
);
GO

Now, let’s populate the Sales table with some sample data:

-- Insert sample data into the Sales table
INSERT INTO Sales (SalesRepID, SalesRepName, SaleDate, SaleAmount) VALUES
(1, 'Alice', '2023-01-15', 1000.00),
(1, 'Alice', '2023-02-15', 1500.00),
(1, 'Alice', '2023-03-15', 1200.00),
(2, 'Bob', '2023-01-20', 800.00),
(2, 'Bob', '2023-02-20', 1600.00),
(2, 'Bob', '2023-03-20', 1100.00),
(3, 'Charlie', '2023-01-25', 1300.00),
(3, 'Charlie', '2023-02-25', 1700.00),
(3, 'Charlie', '2023-03-25', 1800.00);
GO

Improved Support for Ordered Data in Window Functions ๐ŸŒŸ

SQL Server 2022 introduces several enhancements to window functions, making it easier to work with ordered data. Let’s explore these improvements with our use case.

1. Ranking Sales Representatives ๐Ÿ†

To rank sales representatives based on their monthly sales, we can use the RANK() function:

-- Rank sales representatives based on monthly sales
SELECT 
    SalesRepName,
    SaleDate,
    SaleAmount,
    RANK() OVER (PARTITION BY DATEPART(YEAR, SaleDate), DATEPART(MONTH, SaleDate) 
                 ORDER BY SaleAmount DESC) AS SalesRank
FROM 
    Sales
ORDER BY 
    SaleDate, SalesRank;

This query partitions the data by year and month and ranks the sales representatives within each partition based on their sales amount.

2. Calculating Running Total ๐Ÿงฎ

To calculate the running total of sales for each representative, we can use the SUM() function with the ROWS BETWEEN clause:

-- Calculate running total of sales for each representative
SELECT 
    SalesRepName,
    SaleDate,
    SaleAmount,
    SUM(SaleAmount) OVER (PARTITION BY SalesRepID ORDER BY SaleDate 
                          ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW) AS RunningTotal
FROM 
    Sales
ORDER BY 
    SalesRepName, SaleDate;

This query calculates the running total of sales for each representative, ordered by the sale date.

3. Calculating Month-over-Month Difference ๐Ÿ“‰๐Ÿ“ˆ

To determine the difference in sales between the current month and the previous month, we can use the LAG() function:

-- Calculate month-over-month difference in sales
SELECT 
    SalesRepName,
    SaleDate,
    SaleAmount,
    SaleAmount - LAG(SaleAmount, 1, 0) OVER (PARTITION BY SalesRepID ORDER BY SaleDate) AS MonthOverMonthDifference
FROM 
    Sales
ORDER BY 
    SalesRepName, SaleDate;

This query calculates the difference in sales between the current month and the previous month for each sales representative.

4. Average Monthly Sales per Representative ๐Ÿ“Š

To calculate the average monthly sales for each representative:

-- Calculate average monthly sales for each representative
SELECT 
    SalesRepName,
    DATEPART(YEAR, SaleDate) AS SaleYear,
    DATEPART(MONTH, SaleDate) AS SaleMonth,
    AVG(SaleAmount) OVER (PARTITION BY SalesRepID, DATEPART(YEAR, SaleDate), DATEPART(MONTH, SaleDate)) AS AvgMonthlySales
FROM 
    Sales
ORDER BY 
    SalesRepName, SaleYear, SaleMonth;

5. Cumulative Distribution of Sales ๐Ÿ“ˆ

To compute the cumulative distribution of sales amounts within each month:

-- Calculate cumulative distribution of sales within each month
SELECT 
    SalesRepName,
    SaleDate,
    SaleAmount,
    CUME_DIST() OVER (PARTITION BY DATEPART(YEAR, SaleDate), DATEPART(MONTH, SaleDate) 
                      ORDER BY SaleAmount) AS CumulativeDistribution
FROM 
    Sales
ORDER BY 
    SaleDate, SaleAmount;

6. Percentage Rank of Sales Representatives ๐ŸŽฏ

To assign a percentage rank to sales representatives based on their sales amounts:

-- Calculate percentage rank of sales representatives
SELECT 
    SalesRepName,
    SaleDate,
    SaleAmount,
    PERCENT_RANK() OVER (PARTITION BY DATEPART(YEAR, SaleDate), DATEPART(MONTH, SaleDate) 
                         ORDER BY SaleAmount) AS PercentageRank
FROM 
    Sales
ORDER BY 
    SaleDate, PercentageRank;

7. NTILE Function to Divide Sales into Quartiles ๐Ÿชœ

To divide sales amounts into quartiles for better distribution analysis:

-- Divide sales into quartiles
SELECT 
    SalesRepName,
    SaleDate,
    SaleAmount,
    NTILE(4) OVER (PARTITION BY DATEPART(YEAR, SaleDate), DATEPART(MONTH, SaleDate) 
                   ORDER BY SaleAmount) AS SalesQuartile
FROM 
    Sales
ORDER BY 
    SaleDate, SalesQuartile;

8. Median Sale Amount per Month ๐Ÿ“

To calculate the median sale amount for each month using the PERCENTILE_CONT function:

-- Calculate median sale amount per month
SELECT DISTINCT
    DATEPART(YEAR, SaleDate) AS SaleYear,
    DATEPART(MONTH, SaleDate) AS SaleMonth,
    PERCENTILE_CONT(0.5) WITHIN GROUP (ORDER BY SaleAmount) OVER (PARTITION BY DATEPART(YEAR, SaleDate), DATEPART(MONTH, SaleDate)) AS MedianSaleAmount
FROM 
    Sales
ORDER BY 
    SaleYear, SaleMonth;

9. Lead Function to Compare Next Month Sales ๐Ÿ“…

To compare the sales amount with the sales of the next month:

-- Compare sales amount with next month's sales
SELECT 
    SalesRepName,
    SaleDate,
    SaleAmount,
    LEAD(SaleAmount, 1, 0) OVER (PARTITION BY SalesRepID ORDER BY SaleDate) AS NextMonthSales,
    LEAD(SaleAmount, 1, 0) OVER (PARTITION BY SalesRepID ORDER BY SaleDate) - SaleAmount AS SalesDifference
FROM 
    Sales
ORDER BY 
    SalesRepName, SaleDate;

Conclusion ๐ŸŽ‰

SQL Server 2022’s enhanced support for ordered data in window functions provides powerful tools for analyzing and manipulating data. In this blog, we demonstrated how to use these improvements to rank sales representatives, calculate running totals, and determine month-over-month sales differences.

These enhancements simplify complex queries and improve performance, making it easier to gain insights from your data. Whether you’re analyzing sales performance or tackling other business challenges, SQL Server 2022’s window functions can help you achieve your goals more efficiently. ๐ŸŒŸ

Happy querying! ๐Ÿ’ป

For more tutorials and tips on  SQL Server, including performance tuning and  database management, be sure to check out our JBSWiki YouTube channel.

Thank You,
Vivek Janakiraman

Disclaimer:
The views expressed on this blog are mine alone and do not reflect the views of my company or anyone else. All postings on this blog are provided โ€œAS ISโ€ with no warranties, and confers no rights.

Mastering LAG and LEAD Functions in SQL Server 2022 with the IGNORE NULLS Option

SQL Server 2022 introduced a powerful enhancement to the LAG and LEAD functions with the IGNORE NULLS option. This feature allows for more precise analysis and reporting by skipping over NULL values in data sets. In this blog, weโ€™ll explore how to use these functions effectively using the JBDB database, and we’ll demonstrate their application with a detailed business use case.

Business Use Case: Sales Data Analysis

Imagine a retail company, JBStore, that wants to analyze its sales data to understand sales trends better. They aim to compare each month’s sales with the previous and next months, ignoring any missing data (represented by NULL values). This analysis will help identify trends and outliers, aiding in better decision-making.

Setting Up the JBDB Database

First, letโ€™s set up the JBDB database and create a SalesData table with some sample data, including NULL values to represent months with no sales data.

-- Create JBDB database
CREATE DATABASE JBDB;
GO

-- Use the JBDB database
USE JBDB;
GO

-- Create SalesData table
CREATE TABLE SalesData (
    SalesMonth INT,
    SalesAmount INT
);

-- Insert sample data, including NULLs
INSERT INTO SalesData (SalesMonth, SalesAmount)
VALUES
    (1, 1000),
    (2, 1500),
    (3, NULL),
    (4, 1800),
    (5, NULL),
    (6, 2000);
GO

LAG and LEAD Functions: A Quick Recap

The LAG function allows you to access data from a previous row in the same result set without the use of a self-join. Similarly, the LEAD function accesses data from a subsequent row. Both functions are part of the SQL window functions family and are particularly useful in time series analysis.

Using LAG and LEAD with IGNORE NULLS

The IGNORE NULLS option is a game-changer, as it allows you to skip over NULL values, providing more meaningful results. Here’s how you can use it with the LAG and LEAD functions:

Example 1: LAG Function with IGNORE NULLS
SELECT 
    SalesMonth,
    SalesAmount,
    LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS PreviousMonthSales
FROM 
    SalesData;

In this example, LAG(SalesAmount, 1) IGNORE NULLS retrieves the sales amount from the previous month, skipping over any NULL values.

Example 2: LEAD Function with IGNORE NULLS
SELECT 
    SalesMonth,
    SalesAmount,
    LEAD(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS NextMonthSales
FROM 
    SalesData;

Here, LEAD(SalesAmount, 1) IGNORE NULLS retrieves the sales amount from the next month, again skipping over NULL values.

Practical Example: Analyzing Sales Trends

Letโ€™s combine these functions to analyze sales trends more effectively.

SELECT 
    SalesMonth,
    SalesAmount,
    LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS PreviousMonthSales,
    LEAD(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS NextMonthSales
FROM 
    SalesData;

This query provides a complete view of each month’s sales, the previous month’s sales, and the next month’s sales, excluding any NULL values. This is incredibly useful for identifying patterns, such as periods of growth or decline.

Detailed Business Use Case: Data-Driven Decision Making

By utilizing the IGNORE NULLS option with LAG and LEAD functions, JBStore can:

  1. Identify Growth Periods: Detect months where sales increased significantly compared to the previous or next month.
  2. Spot Anomalies: Easily identify months with unusually high or low sales, excluding months with missing data.
  3. Trend Analysis: Understand longer-term trends by comparing sales over multiple months.

These insights can inform marketing strategies, inventory planning, and more.

Calculate Difference Between Current and Previous Month’s Sales:

SELECT SalesMonth, SalesAmount, SalesAmount - LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS SalesDifference FROM SalesData;

Identify Months with Sales Decrease Compared to Previous Month:

WITH CTE AS (
    SELECT 
        SalesMonth,
        SalesAmount,
        LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS PreviousMonthSales
    FROM 
        SalesData
)
SELECT 
    SalesMonth,
    SalesAmount,
    PreviousMonthSales
FROM 
    CTE
WHERE 
    SalesAmount < PreviousMonthSales;

Find the Second Previous Month’s Sales:

SELECT SalesMonth, SalesAmount, LAG(SalesAmount, 2) IGNORE NULLS OVER (ORDER BY SalesMonth) AS SecondPreviousMonthSales FROM SalesData;

Calculate the Rolling Average of the Last Two Months (Ignoring NULLs):

SELECT SalesMonth, SalesAmount, (SalesAmount + LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth)) / 2 AS RollingAverage FROM SalesData;

Compare Sales Between Current Month and Two Months Ahead:

SELECT SalesMonth, SalesAmount, LEAD(SalesAmount, 2) IGNORE NULLS OVER (ORDER BY SalesMonth) AS SalesTwoMonthsAhead FROM SalesData;

Identify Consecutive Months with Sales Increase:

WITH CTE AS ( SELECT SalesMonth, SalesAmount, LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS PreviousMonthSales FROM SalesData ) SELECT SalesMonth, SalesAmount FROM CTE WHERE SalesAmount > PreviousMonthSales;

Find Months with No Sales and Their Preceding Sales Month:

SELECT SalesMonth, SalesAmount, LAG(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS PrecedingMonthSales FROM SalesData WHERE SalesAmount IS NULL;

Calculate Cumulative Sales Sum Ignoring NULLs:

SELECT 
    SalesMonth,
    SalesAmount,
    SUM(ISNULL(SalesAmount, 0)) OVER (ORDER BY SalesMonth ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW) AS CumulativeSales
FROM 
    SalesData;

Identify the First Month with Sales After a Month with NULL Sales:

SELECT SalesMonth, SalesAmount, LEAD(SalesAmount, 1) IGNORE NULLS OVER (ORDER BY SalesMonth) AS FirstNonNullSalesAfterNull FROM SalesData WHERE SalesAmount IS NULL;

    Conclusion ๐ŸŽ‰

    The LAG and LEAD functions with the IGNORE NULLS option in SQL Server 2022 offer a more refined way to analyze data, providing more accurate and meaningful results. Whether you’re analyzing sales data, customer behavior, or any other time series data, these functions can significantly enhance your analytical capabilities.

    Happy querying! ๐Ÿš€

    For more tutorials and tips on SQL Server, including performance tuning and database management, be sure to check out our JBSWiki YouTube channel.

    Thank You,
    Vivek Janakiraman

    Disclaimer:
    The views expressed on this blog are mine alone and do not reflect the views of my company or anyone else. All postings on this blog are provided โ€œAS ISโ€ with no warranties, and confers no rights.