|
@@ -97,16 +97,16 @@ def get_portfolio_weekly_rets(portfolio_ids, start_date, end_date, isFromMySQL)
|
|
|
return t
|
|
|
}
|
|
|
|
|
|
+/*
|
|
|
+ * 取月收益
|
|
|
+ *
|
|
|
+ * Example: get_monthly_ret('FD', ['HF000004KN','HF000103EU','HF00018WXG'], 2000.01.01, 2024.03.01, true);
|
|
|
+ */
|
|
|
def get_monthly_ret(entity_type, entity_ids, start_date, end_date, isFromMySQL) {
|
|
|
|
|
|
- s_entity_ids = '';
|
|
|
-
|
|
|
- // 判断输入的 fund_ids 是字符串标量还是向量
|
|
|
- if ( entity_ids.form() == 0 ) {
|
|
|
- s_entity_ids = entity_ids;
|
|
|
- } else {
|
|
|
- s_entity_ids = "'" + entity_ids.concat("','") + "'";
|
|
|
- }
|
|
|
+ s_entity_ids = ids_to_string(entity_ids);
|
|
|
+
|
|
|
+ if(s_entity_ids == null || s_entity_ids == '') return null;
|
|
|
|
|
|
tmp = get_performance_table_description(entity_type);
|
|
|
|
|
@@ -204,25 +204,21 @@ def get_fund_latest_nav_performance(fund_ids, isFromMySQL) {
|
|
|
}
|
|
|
|
|
|
/*
|
|
|
- * 取私募基金净值
|
|
|
+ * 取基金净值
|
|
|
*
|
|
|
*
|
|
|
* Create: 202408 Joey
|
|
|
* TODO: add isvalid and nav > 0 for local version
|
|
|
*
|
|
|
*
|
|
|
- * Example: get_nav_by_price_date('HF', "'HF000004KN','HF00018WXG'", 2024.05.01, true)
|
|
|
+ * Example: get_nav_by_price_date('HF', "'HF000004KN','HF00018WXG'", 2024.05.01, true);
|
|
|
+ * get_nav_by_price_date('IN', "'IN00000008','IN0000000M'", 2024.05.01, true);
|
|
|
*/
|
|
|
def get_nav_by_price_date(entity_type, entity_ids, price_date, isFromMySQL) {
|
|
|
|
|
|
- s_entity_ids = '';
|
|
|
-
|
|
|
- // 判断输入的 fund_ids 是字符串标量还是向量
|
|
|
- if ( entity_ids.form() == 0 ) {
|
|
|
- s_entity_ids = entity_ids;
|
|
|
- } else {
|
|
|
- s_entity_ids = "'" + entity_ids.concat("','") + "'";
|
|
|
- }
|
|
|
+ s_entity_ids = ids_to_string(entity_ids);
|
|
|
+
|
|
|
+ if(s_entity_ids == null || s_entity_ids == '') return null;
|
|
|
|
|
|
tmp = get_nav_table_description(entity_type);
|
|
|
|
|
@@ -305,15 +301,20 @@ def get_index_nav_by_price_date(index_ids, price_date) {
|
|
|
/*
|
|
|
* 取有效基金基本信息
|
|
|
*
|
|
|
- * Example: get_fund_info("'HF000004KN','HF00018WXG'")
|
|
|
+ * Example: get_fund_info("'HF000004KN','HF00018WXG'");
|
|
|
+ * get_fund_info(['HF000004KN','HF00018WXG']);
|
|
|
*
|
|
|
*/
|
|
|
def get_fund_info(fund_ids) {
|
|
|
|
|
|
+ s_entity_ids = ids_to_string(fund_ids);
|
|
|
+
|
|
|
+ if(s_entity_ids == NULL || s_entity_ids == '') return null;
|
|
|
+
|
|
|
s_query = "SELECT fi.fund_id, fi.inception_date, fi.primary_benchmark_id AS benchmark_id, IFNULL(fi.initial_unit_value, 1) AS ini_value, fs.strategy, fs.substrategy
|
|
|
FROM mfdb.fund_information fi
|
|
|
INNER JOIN mfdb.fund_strategy fs ON fi.fund_id = fs.fund_id AND fs.isvalid = 1
|
|
|
- WHERE fi.fund_id IN (" + fund_ids + ")
|
|
|
+ WHERE fi.fund_id IN (" + s_entity_ids + ")
|
|
|
AND fi.isvalid = 1
|
|
|
ORDER BY fi.fund_id"
|
|
|
|
|
@@ -335,6 +336,10 @@ def get_fund_info(fund_ids) {
|
|
|
*/
|
|
|
def get_portfolio_info(portfolio_ids) {
|
|
|
|
|
|
+ s_entity_ids = ids_to_string(portfolio_ids);
|
|
|
+
|
|
|
+ if(s_entity_ids == NULL || s_entity_ids == '') return null;
|
|
|
+
|
|
|
s_query = "SELECT cpm.id AS portfolio_id, cpm.userid, cpm.customer_id, cpm.inception_date, cpm.portfolio_source, cpm.portfolio_type
|
|
|
FROM pfdb.`pf_customer_portfolio_map` cpm
|
|
|
INNER JOIN pfdb.cm_user u ON cpm.userid = u.userid
|
|
@@ -353,15 +358,47 @@ def get_portfolio_info(portfolio_ids) {
|
|
|
}
|
|
|
|
|
|
/*
|
|
|
- * 取私募基金净值更新信息, 返回基金及其净值更新的最早净值日期
|
|
|
+ * 取基金组合基础有效信息
|
|
|
+ *
|
|
|
+ * Example: get_entity_info('FD', ['HF000004KN','HF000103EU','HF00018WXG']);
|
|
|
+ * get_entity_info('PF', '166002,166114');
|
|
|
+ */
|
|
|
+def get_entity_info(entity_type, entity_ids) {
|
|
|
+
|
|
|
+ t = null;
|
|
|
+
|
|
|
+ s_entity_ids = ids_to_string(entity_ids);
|
|
|
+
|
|
|
+ if(s_entity_ids == null || s_entity_ids == '') return null;
|
|
|
+
|
|
|
+ if(entity_type == 'FD') {
|
|
|
+
|
|
|
+ t = get_fund_info(s_entity_ids);
|
|
|
+
|
|
|
+ t.rename!('fund_id', 'entity_id');
|
|
|
+
|
|
|
+ } else if(entity_type == 'PF') {
|
|
|
+
|
|
|
+ t = get_portfolio_info(s_entity_ids);
|
|
|
+
|
|
|
+ t.rename!('portfolio_id', 'entity_id');
|
|
|
+ }
|
|
|
+
|
|
|
+ return t;
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+/*
|
|
|
+ * 取基金净值更新信息, 返回基金及其净值更新的最早净值日期
|
|
|
*
|
|
|
* @param fund_ids: fund_id STRING VECTOR
|
|
|
* @param update_time: all updates after this time
|
|
|
*
|
|
|
- * Example: get_fund_list_by_nav_updatetime(null, 2024.07.19T10:00:00)
|
|
|
+ * Example: get_fund_list_by_nav_updatetime('MF', null, 2024.09.26);
|
|
|
+ * get_fund_list_by_nav_updatetime('HF', null, 2024.07.19T10:00:00)
|
|
|
*
|
|
|
*/
|
|
|
-def get_fund_list_by_nav_updatetime(fund_ids, updatetime) {
|
|
|
+def get_fund_list_by_nav_updatetime(entity_type, fund_ids, updatetime) {
|
|
|
|
|
|
s_fund_sql = '';
|
|
|
// 这里要用 isVoid, 因为 isNull对向量返回的是布尔向量
|
|
@@ -369,11 +406,17 @@ def get_fund_list_by_nav_updatetime(fund_ids, updatetime) {
|
|
|
s_fund_ids = fund_ids.concat("','");
|
|
|
s_fund_sql = " AND fi.fund_id IN ('" + s_fund_ids + "')";
|
|
|
}
|
|
|
+
|
|
|
+ if(entity_type == 'HF') {
|
|
|
+ nav_table = 'mfdb.nav';
|
|
|
+ } else {
|
|
|
+ nav_table = 'mfdb.public_nav';
|
|
|
+ }
|
|
|
|
|
|
s_query = "SELECT fi.fund_id, MIN(nav.price_date) AS price_date,
|
|
|
fi.inception_date, fi.primary_benchmark_id AS benchmark_id, IFNULL(fi.initial_unit_value, 1) AS ini_value
|
|
|
FROM mfdb.fund_information fi
|
|
|
- INNER JOIN mfdb.nav ON fi.fund_id = nav.fund_id
|
|
|
+ INNER JOIN " + nav_table + " nav ON fi.fund_id = nav.fund_id
|
|
|
WHERE fi.isvalid = 1" +
|
|
|
s_fund_sql + "
|
|
|
AND nav.cumulative_nav > 0
|
|
@@ -471,6 +514,37 @@ def get_portfolio_primary_benchmark(portfolio_ids, month_start, month_end) {
|
|
|
|
|
|
}
|
|
|
|
|
|
+/*
|
|
|
+ * 取某时间段的基金组合主基准
|
|
|
+ *
|
|
|
+ *
|
|
|
+ * Example: get_entity_primary_benchmark('FD', "'MF00003PW2', 'MF00003PW1', 'MF00003PXO'", '1990-01', '2024-06');
|
|
|
+ * get_entity_primary_benchmark('PF', [166002,166114], '1990-01', '2024-08');
|
|
|
+ */
|
|
|
+def get_entity_primary_benchmark(entity_type, entity_ids, month_start, month_end) {
|
|
|
+
|
|
|
+ t = null;
|
|
|
+
|
|
|
+ s_entity_ids = ids_to_string(entity_ids);
|
|
|
+
|
|
|
+ if(s_entity_ids == null || s_entity_ids == '') return null;
|
|
|
+
|
|
|
+ if(entity_type == 'FD') {
|
|
|
+
|
|
|
+ t = get_fund_primary_benchmark(s_entity_ids, month_start, month_end);
|
|
|
+
|
|
|
+ t.rename!('fund_id', 'entity_id');
|
|
|
+
|
|
|
+ } else if(entity_type == 'PF') {
|
|
|
+
|
|
|
+ t = get_portfolio_primary_benchmark(s_entity_ids, month_start, month_end);
|
|
|
+
|
|
|
+ t.rename!('portfolio_id', 'entity_id');
|
|
|
+ }
|
|
|
+
|
|
|
+ return t;
|
|
|
+
|
|
|
+}
|
|
|
|
|
|
/*
|
|
|
* 取某时间段的基金BFI因子
|
|
@@ -523,6 +597,39 @@ def get_portfolio_bfi_factors(portfolio_ids, month_start, month_end) {
|
|
|
|
|
|
}
|
|
|
|
|
|
+
|
|
|
+/*
|
|
|
+ * 取某时间段的基金组合BFI基准
|
|
|
+ *
|
|
|
+ *
|
|
|
+ * Example: get_entity_bfi_factors('FD', "'MF00003PW2', 'MF00003PW1', 'MF00003PXO'", '1990-01', '2024-06');
|
|
|
+ * get_entity_bfi_factors('PF', [166002,166114], '1990-01', '2024-08');
|
|
|
+ */
|
|
|
+def get_entity_bfi_factors(entity_type, entity_ids, month_start, month_end) {
|
|
|
+
|
|
|
+ t = null;
|
|
|
+
|
|
|
+ s_entity_ids = ids_to_string(entity_ids);
|
|
|
+
|
|
|
+ if(s_entity_ids == null || s_entity_ids == '') return null;
|
|
|
+
|
|
|
+ if(entity_type == 'FD') {
|
|
|
+
|
|
|
+ t = get_fund_bfi_factors(s_entity_ids, month_start, month_end);
|
|
|
+
|
|
|
+ t.rename!('fund_id', 'entity_id');
|
|
|
+
|
|
|
+ } else if(entity_type == 'PF') {
|
|
|
+
|
|
|
+ t = get_portfolio_bfi_factors(s_entity_ids, month_start, month_end);
|
|
|
+
|
|
|
+ t.rename!('portfolio_id', 'entity_id');
|
|
|
+ }
|
|
|
+
|
|
|
+ return t;
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
/*
|
|
|
* 取组合交易表
|
|
|
*
|
|
@@ -548,13 +655,16 @@ def get_portfolio_holding_history(portfolio_ids) {
|
|
|
}
|
|
|
|
|
|
/*
|
|
|
- * 取基金证券从某日期后的所有净值
|
|
|
- * @param json_query <JSON>: [{sec_id:xxx, holding_date: yyyy-mm-dd}]
|
|
|
+ * 取基金证券从某日期后的所有净值及前值
|
|
|
+ *
|
|
|
+ * @param entity_type <STRING>: MF, HF, EQ, CF, MI, TI, CI, FA, PF
|
|
|
+ * @param freq <STRING>: m, w, d
|
|
|
+ * @param json_query <JSON>: [{sec_id:xxx, price_date: yyyy-mm-dd}]
|
|
|
*
|
|
|
*/
|
|
|
-def get_holding_nav(json_query) {
|
|
|
+def get_nav_for_return_calculation(entity_type, freq, json_query) {
|
|
|
|
|
|
- s_query = "CALL pfdb.sp_get_nav_after_date('" + json_query + "')";
|
|
|
+ s_query = "CALL pfdb.sp_get_nav_after_date('" + entity_type + "', '" + freq + "', '" + json_query + "')";
|
|
|
|
|
|
conn = connect_mysql();
|
|
|
|
|
@@ -564,3 +674,37 @@ def get_holding_nav(json_query) {
|
|
|
|
|
|
return t;
|
|
|
}
|
|
|
+
|
|
|
+/*
|
|
|
+ * 取主基准和BFI的历史月收益率
|
|
|
+ *
|
|
|
+ * @param benchmarks <TABLE>: entity-benchmark 的对应关系表
|
|
|
+ * @param end_day <DATE>: 收益的截止日期
|
|
|
+ *
|
|
|
+ * @return <TABLE>: benchmark_id, end_date, ret
|
|
|
+ *
|
|
|
+ */
|
|
|
+def get_benchmark_return(benchmarks, end_day) {
|
|
|
+
|
|
|
+ s_index_ids = '';
|
|
|
+ s_factor_ids = '';
|
|
|
+
|
|
|
+ if(benchmarks.isVoid() || benchmarks.size() == 0) { return null; }
|
|
|
+
|
|
|
+ // 前缀为 IN 的 benchmark id
|
|
|
+ t_index_id = SELECT DISTINCT benchmark_id FROM benchmarks WHERE benchmark_id LIKE 'IN%';
|
|
|
+ s_index_ids = iif(isVoid(t_index_id), "", "'" + t_index_id.benchmark_id.concat("','") + "'");
|
|
|
+
|
|
|
+ // 前缀为 FA 的 benchmark id
|
|
|
+ t_factor_id = SELECT DISTINCT benchmark_id FROM benchmarks WHERE benchmark_id LIKE 'FA%';
|
|
|
+ s_factor_ids = iif(isVoid(t_factor_id), "", "'" + t_factor_id.benchmark_id.concat("','") + "'");
|
|
|
+
|
|
|
+
|
|
|
+ // 目前指数的月度业绩存在 fund_performance 表
|
|
|
+ t_bmk = SELECT fund_id AS benchmark_id, temporalParse(end_date, 'yyyy-MM') AS end_date, ret FROM get_monthly_ret('IX', s_index_ids, 1990.01.01, end_day, true);
|
|
|
+
|
|
|
+ // 而因子的月度业绩存在 cm_factor_performance 表
|
|
|
+ INSERT INTO t_bmk SELECT factor_id, temporalParse(end_date, 'yyyy-MM') AS end_date, ret FROM get_monthly_ret('FA', s_factor_ids, 1990.01.01, end_day, true);
|
|
|
+
|
|
|
+ return t_bmk;
|
|
|
+}
|