2011-02-17 01:26:22 +00:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Net;
|
2011-04-07 04:40:56 +00:00
|
|
|
using Android.Runtime;
|
2011-02-17 01:26:22 +00:00
|
|
|
using Android.Util;
|
2011-02-26 15:13:52 +00:00
|
|
|
using LumenWorks.Framework.IO.Csv;
|
2011-02-17 01:26:22 +00:00
|
|
|
using MonoStockPortfolio.Entities;
|
|
|
|
|
|
|
|
namespace MonoStockPortfolio.Core.StockData
|
|
|
|
{
|
2011-04-07 04:40:56 +00:00
|
|
|
[Preserve(AllMembers = true)]
|
2011-02-17 01:26:22 +00:00
|
|
|
public class YahooStockDataProvider : IStockDataProvider
|
|
|
|
{
|
|
|
|
private const string LAST_TRADE_PRICE_ONLY = "l1";
|
|
|
|
private const string NAME = "n";
|
|
|
|
private const string VOLUME = "v";
|
|
|
|
private const string TICKER_SYMBOL = "s";
|
|
|
|
private const string CHANGE = "c1";
|
|
|
|
private const string LAST_TRADE_TIME = "t1";
|
|
|
|
private const string REAL_TIME_LAST_TRADE_WITH_TIME = "k1";
|
|
|
|
private const string REAL_TIME_CHANGE = "c6";
|
|
|
|
|
|
|
|
// http://www.gummy-stuff.org/Yahoo-data.htm
|
|
|
|
// http://finance.yahoo.com/d/quotes.csv?s= a BUNCH of
|
|
|
|
// STOCK SYMBOLS separated by "+" &f=a bunch of special tags
|
|
|
|
public IEnumerable<StockQuote> GetStockQuotes(IEnumerable<string> tickers)
|
|
|
|
{
|
|
|
|
string url = "http://finance.yahoo.com/d/quotes.csv?s=";
|
|
|
|
url += string.Join("+", tickers.ToArray());
|
|
|
|
url += "&f=";
|
2011-02-26 16:06:43 +00:00
|
|
|
|
|
|
|
// the ordering is important, if it's changed here, it should be changed
|
|
|
|
// in the ParseCsvIntoStockQuotes method too
|
2011-02-17 01:26:22 +00:00
|
|
|
url += TICKER_SYMBOL;
|
|
|
|
url += LAST_TRADE_PRICE_ONLY;
|
|
|
|
url += NAME;
|
|
|
|
url += VOLUME;
|
|
|
|
url += CHANGE;
|
|
|
|
url += LAST_TRADE_TIME;
|
|
|
|
url += REAL_TIME_LAST_TRADE_WITH_TIME;
|
|
|
|
url += REAL_TIME_CHANGE;
|
|
|
|
|
|
|
|
string resultCsv = ScrapeUrl(url);
|
|
|
|
|
|
|
|
var yahooQuoteData = ParseCsvIntoStockQuotes(resultCsv);
|
|
|
|
|
2011-02-26 04:44:01 +00:00
|
|
|
return yahooQuoteData.Select(MapYahooData);
|
2011-02-17 01:26:22 +00:00
|
|
|
}
|
|
|
|
|
2011-02-26 16:27:52 +00:00
|
|
|
// Yahoo API will return a stock with price of 0.00
|
|
|
|
// if it can't find the ticker
|
|
|
|
public bool IsValidTicker(string ticker)
|
|
|
|
{
|
2011-04-02 04:19:55 +00:00
|
|
|
if(string.IsNullOrEmpty(ticker))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2011-02-26 16:27:52 +00:00
|
|
|
var quote = GetStockQuotes(new[] {ticker}).Single();
|
|
|
|
return quote.LastTradePrice > 0.0M;
|
|
|
|
}
|
|
|
|
|
2011-02-17 01:26:22 +00:00
|
|
|
private static StockQuote MapYahooData(YahooFinanceStockData data)
|
|
|
|
{
|
|
|
|
if (data == null)
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
var stock = new StockQuote();
|
|
|
|
stock.Name = data.Name;
|
|
|
|
stock.LastTradePrice = data.LastTradePrice;
|
|
|
|
stock.Ticker = data.Ticker;
|
|
|
|
stock.Volume = data.Volume;
|
|
|
|
stock.Change = data.Change;
|
|
|
|
stock.LastTradeTime = data.LastTradeTime;
|
|
|
|
stock.RealTimeLastTradePrice = decimal.Parse(data.RealTimeLastTradeWithTime
|
|
|
|
.Replace("<b>", "")
|
|
|
|
.Replace("</b>", "")
|
|
|
|
.Replace("N/A -", "")
|
|
|
|
.Trim()
|
|
|
|
);
|
|
|
|
stock.ChangeRealTime = data.ChangeRealTime;
|
|
|
|
return stock;
|
|
|
|
}
|
|
|
|
|
2011-02-26 15:13:52 +00:00
|
|
|
private static IEnumerable<YahooFinanceStockData> ParseCsvIntoStockQuotes(string csv)
|
2011-02-17 01:26:22 +00:00
|
|
|
{
|
2011-02-26 15:13:52 +00:00
|
|
|
using(var csvReader = new CsvReader(new StringReader(csv),false))
|
2011-02-17 01:26:22 +00:00
|
|
|
{
|
2011-02-26 15:13:52 +00:00
|
|
|
while(csvReader.ReadNextRecord())
|
|
|
|
{
|
|
|
|
var d = new YahooFinanceStockData();
|
|
|
|
d.Ticker = csvReader[0];
|
2011-02-26 16:27:52 +00:00
|
|
|
decimal.TryParse(csvReader[1], out d.LastTradePrice);
|
|
|
|
//d.LastTradePrice = decimal.Parse(csvReader[1]);
|
2011-02-26 15:13:52 +00:00
|
|
|
d.Name = csvReader[2];
|
|
|
|
d.Volume = csvReader[3];
|
2011-02-26 16:27:52 +00:00
|
|
|
decimal.TryParse(csvReader[4], out d.Change);
|
|
|
|
//d.Change = decimal.Parse(csvReader[4]);
|
2011-02-26 15:13:52 +00:00
|
|
|
d.LastTradeTime = csvReader[5];
|
|
|
|
d.RealTimeLastTradeWithTime = csvReader[6];
|
|
|
|
d.ChangeRealTime = csvReader[7];
|
|
|
|
yield return d;
|
|
|
|
}
|
2011-02-17 01:26:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static string ScrapeUrl(string url)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
string resultCsv;
|
|
|
|
var req = WebRequest.Create(url);
|
|
|
|
var resp = req.GetResponse();
|
|
|
|
using (var sr = new StreamReader(resp.GetResponseStream()))
|
|
|
|
{
|
|
|
|
resultCsv = sr.ReadToEnd();
|
|
|
|
sr.Close();
|
|
|
|
}
|
|
|
|
return resultCsv;
|
|
|
|
}
|
|
|
|
catch (Exception ex)
|
|
|
|
{
|
|
|
|
Log.Error("ScrapeUrlException", ex.ToString());
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|